diff options
Diffstat (limited to 'src/game')
-rw-r--r-- | src/game/bg_pmove.c | 8 | ||||
-rw-r--r-- | src/game/g_active.c.orig | 1937 | ||||
-rw-r--r-- | src/game/g_client.c.orig | 1600 | ||||
-rw-r--r-- | src/game/g_cmds.c.orig | 3588 | ||||
-rw-r--r-- | src/game/g_combat.c.orig | 1468 | ||||
-rw-r--r-- | src/game/g_main.c.orig | 2548 |
6 files changed, 7 insertions, 11142 deletions
diff --git a/src/game/bg_pmove.c b/src/game/bg_pmove.c index 3ab4175..3b53dc9 100644 --- a/src/game/bg_pmove.c +++ b/src/game/bg_pmove.c @@ -59,7 +59,12 @@ float cpm_pm_airstopaccelerate = 2.5; //It doesn't need it though, but may feel akward float cpm_pm_jump_z = 0.5; //CPM: 100/270 (normal jumpvel is 270, doublejump default 100) = 0.37037 //However, that value feels too low and nothing much is acomplished since - //tremulous human jump default is only 240 ups, meaning half hight. + //tremulous human jump default is only 240 ups, meaning half height. + +//Clip time for maintaining velocity when clipping. For marauder it's actually equal to the jump +//repeat rate so we can't modify it to cpm's default. CPM Default is 400 I think, but I placed 200 +//becausemarauder has 200, it's a bit tricky to get used to two different values at once. +float pm_cliptime = 200; int c_pmove = 0; @@ -879,6 +884,7 @@ static qboolean PM_CheckJump( void ) //Adding requires me to multiply by class vel again } pm->ps->persistant[PERS_JUMPTIME] = 400; + pm->ps->pm_time = pm_cliptime; //clip through walls with the same timer as walljump } pm->ps->groundEntityNum = ENTITYNUM_NONE; diff --git a/src/game/g_active.c.orig b/src/game/g_active.c.orig deleted file mode 100644 index cfa6f89..0000000 --- a/src/game/g_active.c.orig +++ /dev/null @@ -1,1937 +0,0 @@ -/* -=========================================================================== -Copyright (C) 1999-2005 Id Software, Inc. -Copyright (C) 2000-2009 Darklegion Development - -This file is part of Tremulous. - -Tremulous 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. - -Tremulous is distributed in the hope that it will be -useful, but WITHOUT ANY WARRANTY; without even the implied warranty of -MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -GNU General Public License for more details. - -You should have received a copy of the GNU General Public License -along with Tremulous; if not, write to the Free Software -Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA -=========================================================================== -*/ - -#include "g_local.h" - -/* To create useful and interesting cuboids (like ladders), the ability to specify - * custom surface flags for buildings was needed. This is what the following function - * was written for. It replaces the regular tracing function in PM code (also on the - * client side, grep for CG_PMTraceHack). This function also takes into account - * the modelindex of the entity it found and adjusts the surfaceFlags output - * accordingly. This way custom surface flags for buildings can be done without - * messing around with the PM code or the engine itself. - */ -void G_PMTraceHack(trace_t *results,const vec3_t start,const vec3_t mins,const vec3_t maxs,const vec3_t end,int passEntityNum,int contentmask ) -{ - int modelindex; - trap_Trace(results,start,mins,maxs,end,passEntityNum,contentmask); - if(results->entityNum!=ENTITYNUM_NONE) - { - modelindex=g_entities[results->entityNum].s.modelindex; - if(modelindex>=CUBOID_FIRST&&modelindex<=CUBOID_LAST&&(g_entities[results->entityNum].s.eFlags&EF_B_SPAWNED)) - results->surfaceFlags|=BG_CuboidAttributes(modelindex)->surfaceFlags; - } -} - - -/* -=============== -G_DamageFeedback - -Called just before a snapshot is sent to the given player. -Totals up all damage and generates both the player_state_t -damage values to that client for pain blends and kicks, and -global pain sound events for all clients. -=============== -*/ -void P_DamageFeedback( gentity_t *player ) -{ - gclient_t *client; - float count; - vec3_t angles; - - client = player->client; - if( !PM_Live( client->ps.pm_type ) ) - return; - - // total points of damage shot at the player this frame - count = client->damage_blood + client->damage_armor; - if( count == 0 ) - return; // didn't take any damage - - if( count > 255 ) - count = 255; - - // send the information to the client - - // world damage (falling, slime, etc) uses a special code - // to make the blend blob centered instead of positional - if( client->damage_fromWorld ) - { - client->ps.damagePitch = 255; - client->ps.damageYaw = 255; - - client->damage_fromWorld = qfalse; - } - else - { - vectoangles( client->damage_from, angles ); - client->ps.damagePitch = angles[ PITCH ] / 360.0 * 256; - client->ps.damageYaw = angles[ YAW ] / 360.0 * 256; - } - - // play an apropriate pain sound - if( ( level.time > player->pain_debounce_time ) && !( player->flags & FL_GODMODE ) ) - { - player->pain_debounce_time = level.time + 700; - G_AddEvent( player, EV_PAIN, player->health > 255 ? 255 : player->health ); - client->ps.damageEvent++; - } - - - client->ps.damageCount = count; - - // - // clear totals - // - client->damage_blood = 0; - client->damage_armor = 0; - client->damage_knockback = 0; -} - - - -/* -============= -P_WorldEffects - -Check for lava / slime contents and drowning -============= -*/ -void P_WorldEffects( gentity_t *ent ) -{ - int waterlevel; - - if( ent->client->noclip ) - { - ent->client->airOutTime = level.time + 12000; // don't need air - return; - } - - waterlevel = ent->waterlevel; - - // - // check for drowning - // - if( waterlevel == 3 ) - { - // if out of air, start drowning - if( ent->client->airOutTime < level.time) - { - // drown! - ent->client->airOutTime += 1000; - if( ent->health > 0 ) - { - // take more damage the longer underwater - ent->damage += 2; - if( ent->damage > 15 ) - ent->damage = 15; - - // play a gurp sound instead of a normal pain sound - if( ent->health <= ent->damage ) - G_Sound( ent, CHAN_VOICE, G_SoundIndex( "*drown.wav" ) ); - else if( rand( ) < RAND_MAX / 2 + 1 ) - G_Sound( ent, CHAN_VOICE, G_SoundIndex( "sound/player/gurp1.wav" ) ); - else - G_Sound( ent, CHAN_VOICE, G_SoundIndex( "sound/player/gurp2.wav" ) ); - - // don't play a normal pain sound - ent->pain_debounce_time = level.time + 200; - - G_Damage( ent, NULL, NULL, NULL, NULL, - ent->damage, DAMAGE_NO_ARMOR, MOD_WATER ); - } - } - } - else - { - ent->client->airOutTime = level.time + 12000; - ent->damage = 2; - } - - // - // check for sizzle damage (move to pmove?) - // - if( waterlevel && - ( ent->watertype & ( CONTENTS_LAVA | CONTENTS_SLIME ) ) ) - { - if( ent->health > 0 && - ent->pain_debounce_time <= level.time ) - { - if( ent->watertype & CONTENTS_LAVA ) - { - G_Damage( ent, NULL, NULL, NULL, NULL, - 30 * waterlevel, 0, MOD_LAVA ); - } - - if( ent->watertype & CONTENTS_SLIME ) - { - G_Damage( ent, NULL, NULL, NULL, NULL, - 10 * waterlevel, 0, MOD_SLIME ); - } - } - } -} - - - -/* -=============== -G_SetClientSound -=============== -*/ -void G_SetClientSound( gentity_t *ent ) -{ - if( ent->waterlevel && ( ent->watertype & ( CONTENTS_LAVA | CONTENTS_SLIME ) ) ) - ent->client->ps.loopSound = level.snd_fry; - else - ent->client->ps.loopSound = 0; -} - - - -//============================================================== - -/* -============== -ClientShove -============== -*/ -static int GetClientMass( gentity_t *ent ) -{ - int entMass = 100; - - if( ent->client->pers.teamSelection == TEAM_ALIENS ) - entMass = BG_Class( ent->client->pers.classSelection )->health; - else if( ent->client->pers.teamSelection == TEAM_HUMANS ) - { - if( BG_InventoryContainsUpgrade( UP_BATTLESUIT, ent->client->ps.stats ) ) - entMass *= 2; - } - else - return 0; - return entMass; -} - -static void ClientShove( gentity_t *ent, gentity_t *victim ) -{ - vec3_t dir, push; - float force; - int entMass, vicMass; - - // Don't push if the entity is not trying to move - if( !ent->client->pers.cmd.rightmove && !ent->client->pers.cmd.forwardmove && - !ent->client->pers.cmd.upmove ) - return; - - // Cannot push enemy players unless they are walking on the player - if( !OnSameTeam( ent, victim ) && - victim->client->ps.groundEntityNum != ent - g_entities ) - return; - - // Shove force is scaled by relative mass - entMass = GetClientMass( ent ); - vicMass = GetClientMass( victim ); - if( vicMass <= 0 || entMass <= 0 ) - return; - force = g_shove.value * entMass / vicMass; - if( force < 0 ) - force = 0; - if( force > 150 ) - force = 150; - - // Give the victim some shove velocity - VectorSubtract( victim->r.currentOrigin, ent->r.currentOrigin, dir ); - VectorNormalizeFast( dir ); - VectorScale( dir, force, push ); - VectorAdd( victim->client->ps.velocity, push, victim->client->ps.velocity ); - - // Set the pmove timer so that the other client can't cancel - // out the movement immediately - if( !victim->client->ps.pm_time ) - { - int time; - - time = force * 2 + 0.5f; - if( time < 50 ) - time = 50; - if( time > 200 ) - time = 200; - victim->client->ps.pm_time = time; - victim->client->ps.pm_flags |= PMF_TIME_KNOCKBACK; - } -} - -/* -============== -ClientImpacts -============== -*/ -void ClientImpacts( gentity_t *ent, pmove_t *pm ) -{ - int i; - trace_t trace; - gentity_t *other; - - // clear a fake trace struct for touch function - memset( &trace, 0, sizeof( trace ) ); - - for( i = 0; i < pm->numtouch; i++ ) - { - other = &g_entities[ pm->touchents[ i ] ]; - - // see G_UnlaggedDetectCollisions(), this is the inverse of that. - // if our movement is blocked by another player's real position, - // don't use the unlagged position for them because they are - // blocking or server-side Pmove() from reaching it - if( other->client && other->client->unlaggedCalc.used ) - other->client->unlaggedCalc.used = qfalse; - - // tyrant impact attacks - if( ent->client->ps.weapon == WP_ALEVEL4 ) - { - G_ChargeAttack( ent, other ); - G_CrushAttack( ent, other ); - } - - // shove players - if( ent->client && other->client ) - ClientShove( ent, other ); - - // touch triggers - if( other->touch ) - other->touch( other, ent, &trace ); - } -} - -/* -============ -G_TouchTriggers - -Find all trigger entities that ent's current position touches. -Spectators will only interact with teleporters. -============ -*/ -void G_TouchTriggers( gentity_t *ent ) -{ - int i, num; - int touch[MAX_GENTITIES]; - gentity_t *hit; - trace_t trace; - vec3_t mins, maxs; - vec3_t pmins, pmaxs; - static vec3_t range = { 10, 10, 10 }; - - if( !ent->client ) - return; - - // dead clients don't activate triggers! - if( ent->client->ps.stats[ STAT_HEALTH ] <= 0 ) - return; - - BG_ClassBoundingBox( ent->client->ps.stats[ STAT_CLASS ], - pmins, pmaxs, NULL, NULL, NULL ); - - VectorAdd( ent->client->ps.origin, pmins, mins ); - VectorAdd( ent->client->ps.origin, pmaxs, maxs ); - - VectorSubtract( mins, range, mins ); - VectorAdd( maxs, range, maxs ); - - num = trap_EntitiesInBox( mins, maxs, touch, MAX_GENTITIES ); - - // can't use ent->absmin, because that has a one unit pad - VectorAdd( ent->client->ps.origin, ent->r.mins, mins ); - VectorAdd( ent->client->ps.origin, ent->r.maxs, maxs ); - - for( i = 0; i < num; i++ ) - { - hit = &g_entities[ touch[ i ] ]; - - if( !hit->touch && !ent->touch ) - continue; - - if( !( hit->r.contents & CONTENTS_TRIGGER ) ) - continue; - - // ignore most entities if a spectator - if( ent->client->sess.spectatorState != SPECTATOR_NOT ) - { - if( hit->s.eType != ET_TELEPORT_TRIGGER && - // this is ugly but adding a new ET_? type will - // most likely cause network incompatibilities - hit->touch != Touch_DoorTrigger ) - { - //check for manually triggered doors - manualTriggerSpectator( hit, ent ); - continue; - } - } - - if( !trap_EntityContact( mins, maxs, hit ) ) - continue; - - memset( &trace, 0, sizeof( trace ) ); - - if( hit->touch ) - hit->touch( hit, ent, &trace ); - } -} - -/* -================= -SpectatorThink -================= -*/ -void SpectatorThink( gentity_t *ent, usercmd_t *ucmd ) -{ - pmove_t pm; - gclient_t *client; - int clientNum; - qboolean attack1, attack3, following, queued; - - client = ent->client; - - client->oldbuttons = client->buttons; - client->buttons = ucmd->buttons; - - attack1 = ( client->buttons & BUTTON_ATTACK ) && - !( client->oldbuttons & BUTTON_ATTACK ); - attack3 = ( client->buttons & BUTTON_USE_HOLDABLE ) && - !( client->oldbuttons & BUTTON_USE_HOLDABLE ); - - // We are in following mode only if we are following a non-spectating client - following = client->sess.spectatorState == SPECTATOR_FOLLOW; - if( following ) - { - clientNum = client->sess.spectatorClient; - if( clientNum < 0 || clientNum > level.maxclients || - !g_entities[ clientNum ].client || - g_entities[ clientNum ].client->sess.spectatorState != SPECTATOR_NOT ) - following = qfalse; - } - - // Check to see if we are in the spawn queue - if( client->pers.teamSelection == TEAM_ALIENS ) - queued = G_SearchSpawnQueue( &level.alienSpawnQueue, ent - g_entities ); - else if( client->pers.teamSelection == TEAM_HUMANS ) - queued = G_SearchSpawnQueue( &level.humanSpawnQueue, ent - g_entities ); - else - queued = qfalse; - - // Wants to get out of spawn queue - if( attack1 && queued ) - { - if( client->sess.spectatorState == SPECTATOR_FOLLOW ) - G_StopFollowing( ent ); - if( client->ps.stats[ STAT_TEAM ] == TEAM_ALIENS ) - G_RemoveFromSpawnQueue( &level.alienSpawnQueue, client->ps.clientNum ); - else if( client->ps.stats[ STAT_TEAM ] == TEAM_HUMANS ) - G_RemoveFromSpawnQueue( &level.humanSpawnQueue, client->ps.clientNum ); - client->pers.classSelection = PCL_NONE; - client->ps.stats[ STAT_CLASS ] = PCL_NONE; - client->ps.pm_flags &= ~PMF_QUEUED; - queued = qfalse; - } - else if( attack1 ) - { - // Wants to get into spawn queue - if( client->sess.spectatorState == SPECTATOR_FOLLOW ) - G_StopFollowing( ent ); - if( client->pers.teamSelection == TEAM_NONE ) - G_TriggerMenu( client->ps.clientNum, MN_TEAM ); - else if( client->pers.teamSelection == TEAM_ALIENS ) - G_TriggerMenu( client->ps.clientNum, MN_A_CLASS ); - else if( client->pers.teamSelection == TEAM_HUMANS ) - G_TriggerMenu( client->ps.clientNum, MN_H_SPAWN ); - } - - // We are either not following anyone or following a spectator - if( !following ) - { - if( client->sess.spectatorState == SPECTATOR_LOCKED || - client->sess.spectatorState == SPECTATOR_FOLLOW ) - client->ps.pm_type = PM_FREEZE; - else if( client->noclip ) - client->ps.pm_type = PM_NOCLIP; - else - client->ps.pm_type = PM_SPECTATOR; - - if( queued ) - client->ps.pm_flags |= PMF_QUEUED; - - client->ps.speed = client->pers.flySpeed; - client->ps.stats[ STAT_STAMINA ] = 0; - client->ps.stats[ STAT_MISC ] = 0; - client->buildTimer=0; - client->ps.stats[ STAT_BUILDABLE ] = BA_NONE; - client->ps.stats[ STAT_CLASS ] = PCL_NONE; - client->ps.weapon = WP_NONE; - - // Set up for pmove - memset( &pm, 0, sizeof( pm ) ); - pm.ps = &client->ps; - pm.pmext = &client->pmext; - pm.cmd = *ucmd; - pm.tracemask = MASK_DEADSOLID; // spectators can fly through bodies - pm.trace = G_PMTraceHack; - pm.pointcontents = trap_PointContents; - - // Perform a pmove - Pmove( &pm ); - - // Save results of pmove - VectorCopy( client->ps.origin, ent->s.origin ); - - G_TouchTriggers( ent ); - trap_UnlinkEntity( ent ); - - // Set the queue position and spawn count for the client side - if( client->ps.pm_flags & PMF_QUEUED ) - { - if( client->ps.stats[ STAT_TEAM ] == TEAM_ALIENS ) - { - client->ps.persistant[ PERS_QUEUEPOS ] = - G_GetPosInSpawnQueue( &level.alienSpawnQueue, client->ps.clientNum ); - client->ps.persistant[ PERS_SPAWNS ] = level.numAlienSpawns; - } - else if( client->ps.stats[ STAT_TEAM ] == TEAM_HUMANS ) - { - client->ps.persistant[ PERS_QUEUEPOS ] = - G_GetPosInSpawnQueue( &level.humanSpawnQueue, client->ps.clientNum ); - client->ps.persistant[ PERS_SPAWNS ] = level.numHumanSpawns; - } - } - } -} - - - -/* -================= -ClientInactivityTimer - -Returns qfalse if the client is dropped -================= -*/ -qboolean ClientInactivityTimer( gentity_t *ent ) -{ - gclient_t *client = ent->client; - - if( ! g_inactivity.integer ) - { - // give everyone some time, so if the operator sets g_inactivity during - // gameplay, everyone isn't kicked - client->inactivityTime = level.time + 60 * 1000; - client->inactivityWarning = qfalse; - } - else if( client->pers.cmd.forwardmove || - client->pers.cmd.rightmove || - client->pers.cmd.upmove || - ( client->pers.cmd.buttons & BUTTON_ATTACK ) ) - { - client->inactivityTime = level.time + g_inactivity.integer * 1000; - client->inactivityWarning = qfalse; - } - else if( !client->pers.localClient ) - { - if( level.time > client->inactivityTime && - !G_admin_permission( ent, ADMF_ACTIVITY ) ) - { - trap_DropClient( client - level.clients, "Dropped due to inactivity" ); - return qfalse; - } - - if( level.time > client->inactivityTime - 10000 && - !client->inactivityWarning && - !G_admin_permission( ent, ADMF_ACTIVITY ) ) - { - client->inactivityWarning = qtrue; - trap_SendServerCommand( client - level.clients, "cp \"Ten seconds until inactivity drop!\n\"" ); - } - } - - return qtrue; -} - -/* -================== -ClientTimerActions - -Actions that happen once a second -================== -*/ -void ClientTimerActions( gentity_t *ent, int msec ) -{ - gclient_t *client; - usercmd_t *ucmd; - int aForward, aRight; - qboolean walking = qfalse, stopped = qfalse, - crouched = qfalse, jumping = qfalse, - strafing = qfalse; - int i; - - ucmd = &ent->client->pers.cmd; - - aForward = abs( ucmd->forwardmove ); - aRight = abs( ucmd->rightmove ); - - if( aForward == 0 && aRight == 0 ) - stopped = qtrue; - else if( aForward <= 64 && aRight <= 64 ) - walking = qtrue; - - if( aRight > 0 ) - strafing = qtrue; - - if( ucmd->upmove > 0 ) - jumping = qtrue; - else if( ent->client->ps.pm_flags & PMF_DUCKED ) - crouched = qtrue; - - client = ent->client; - client->time100 += msec; - client->time1000 += msec; - client->time10000 += msec; - - while ( client->time100 >= 100 ) - { - weapon_t weapon = BG_GetPlayerWeapon( &client->ps ); - - client->time100 -= 100; - - // Restore or subtract stamina - if( stopped || client->ps.pm_type == PM_JETPACK ) - client->ps.stats[ STAT_STAMINA ] += STAMINA_STOP_RESTORE; - else if( ( client->ps.stats[ STAT_STATE ] & SS_SPEEDBOOST ) && - !( client->buttons & BUTTON_WALKING ) ) // walk overrides sprint - client->ps.stats[ STAT_STAMINA ] -= STAMINA_SPRINT_TAKE; - else if( walking || crouched ) - client->ps.stats[ STAT_STAMINA ] += STAMINA_WALK_RESTORE; - - // Check stamina limits - if( client->ps.stats[ STAT_STAMINA ] > STAMINA_MAX ) - client->ps.stats[ STAT_STAMINA ] = STAMINA_MAX; - else if( client->ps.stats[ STAT_STAMINA ] < -STAMINA_MAX ) - client->ps.stats[ STAT_STAMINA ] = -STAMINA_MAX; - - if( weapon == WP_ABUILD || weapon == WP_ABUILD2 || - BG_InventoryContainsWeapon( WP_HBUILD, client->ps.stats ) ) - { - client->buildTimer-=MIN(100,client->buildTimer); - G_RecalcBuildTimer(client); - } - - switch( weapon ) - { - case WP_ABUILD: - case WP_ABUILD2: - case WP_HBUILD: - - // Set validity bit on buildable - if( ( client->ps.stats[ STAT_BUILDABLE ] & ~SB_VALID_TOGGLEBIT ) > BA_NONE ) - { - int dist = BG_Class( ent->client->ps.stats[ STAT_CLASS ] )->buildDist; - vec3_t dummy, dummy2; - - if( G_CanBuild( ent, client->ps.stats[ STAT_BUILDABLE ] & ~SB_VALID_TOGGLEBIT, - dist, dummy, dummy2, client->cuboidSelection ) == IBE_NONE ) - client->ps.stats[ STAT_BUILDABLE ] |= SB_VALID_TOGGLEBIT; - else - client->ps.stats[ STAT_BUILDABLE ] &= ~SB_VALID_TOGGLEBIT; - - // Let the client know which buildables will be removed by building - for( i = 0; i < MAX_MISC; i++ ) - { - if( i < level.numBuildablesForRemoval ) - client->ps.misc[ i ] = level.markedBuildables[ i ]->s.number; - else - client->ps.misc[ i ] = 0; - } - } - else - { - for( i = 0; i < MAX_MISC; i++ ) - client->ps.misc[ i ] = 0; - } - break; - - default: - break; - } - - if( ent->client->pers.teamSelection == TEAM_HUMANS && - ( client->ps.stats[ STAT_STATE ] & SS_HEALING_2X ) ) - { - int remainingStartupTime = MEDKIT_STARTUP_TIME - ( level.time - client->lastMedKitTime ); - - if( remainingStartupTime < 0 ) - { - if( ent->health < ent->client->ps.stats[ STAT_MAX_HEALTH ] && - ent->client->medKitHealthToRestore && - ent->client->ps.pm_type != PM_DEAD ) - { - ent->client->medKitHealthToRestore--; - ent->health++; - } - else - ent->client->ps.stats[ STAT_STATE ] &= ~SS_HEALING_2X; - } - else - { - if( ent->health < ent->client->ps.stats[ STAT_MAX_HEALTH ] && - ent->client->medKitHealthToRestore && - ent->client->ps.pm_type != PM_DEAD ) - { - //partial increase - if( level.time > client->medKitIncrementTime ) - { - ent->client->medKitHealthToRestore--; - ent->health++; - - client->medKitIncrementTime = level.time + - ( remainingStartupTime / MEDKIT_STARTUP_SPEED ); - } - } - else - ent->client->ps.stats[ STAT_STATE ] &= ~SS_HEALING_2X; - } - } - - //use fuel - if( BG_InventoryContainsUpgrade( UP_JETPACK, ent->client->ps.stats ) && - BG_UpgradeIsActive( UP_JETPACK, ent->client->ps.stats ) && - ent->client->ps.stats[ STAT_FUEL ] > 0 ) - { - ent->client->ps.stats[ STAT_FUEL ] -= JETPACK_FUEL_USAGE; - if( ent->client->ps.stats[ STAT_FUEL ] <= 0 ) - { - ent->client->ps.stats[ STAT_FUEL ] = 0; - BG_DeactivateUpgrade( UP_JETPACK, client->ps.stats ); - BG_AddPredictableEventToPlayerstate( EV_JETPACK_DEACTIVATE, 0, &client->ps ); - } - } - - //biores - if( BG_InventoryContainsUpgrade( UP_BIORES, ent->client->ps.stats ) && - ent->health < ent->client->ps.stats[ STAT_MAX_HEALTH ] ) - { - float hp, dmod; - - dmod = MIN( level.time - ent->lastDamageTime, 3000 ) / 3000.0f; - hp = (float)ent->health / ent->client->ps.stats[ STAT_MAX_HEALTH ]; - ent->client->bioresHealTimer += (BIORES_EQUATION) * 10 * dmod; - } - - if( ent->client->bioresHealTimer >= 100 ) - { - int delta; - - delta = ent->client->bioresHealTimer / 100; - ent->health = MAX( MIN( ent->health+delta, ent->client->ps.stats[ STAT_MAX_HEALTH ] ), 1 ); - - ent->client->bioresHealTimer %= 100; - } - - } - - while( client->time1000 >= 1000 ) - { - client->time1000 -= 1000; - - //client is poisoned - if( client->ps.stats[ STAT_STATE ] & SS_POISONED ) - { - int damage = ALIEN_POISON_DMG; - - if( BG_InventoryContainsUpgrade( UP_BATTLESUIT, client->ps.stats ) ) - damage -= BSUIT_POISON_PROTECTION; - - if( BG_InventoryContainsUpgrade( UP_HELMET_MK1, client->ps.stats ) ) - damage -= HELMET_MK1_POISON_PROTECTION; - - if( BG_InventoryContainsUpgrade( UP_HELMET_MK2, client->ps.stats ) ) - damage -= HELMET_MK2_POISON_PROTECTION; - - if( BG_InventoryContainsUpgrade( UP_LIGHTARMOUR, client->ps.stats ) ) - damage -= LIGHTARMOUR_POISON_PROTECTION; - - if( BG_InventoryContainsUpgrade( UP_BIORES, client->ps.stats ) ) - damage *= BIORES_POISON_MODIFIER; - - G_Damage( ent, client->lastPoisonClient, client->lastPoisonClient, NULL, - 0, damage, 0, MOD_POISON ); - } - - // turn off life support when a team admits defeat - if( client->ps.stats[ STAT_TEAM ] == TEAM_ALIENS && - level.surrenderTeam == TEAM_ALIENS ) - { - G_Damage( ent, NULL, NULL, NULL, NULL, - BG_Class( client->ps.stats[ STAT_CLASS ] )->regenRate, - DAMAGE_NO_ARMOR, MOD_SUICIDE ); - } - else if( client->ps.stats[ STAT_TEAM ] == TEAM_HUMANS && - level.surrenderTeam == TEAM_HUMANS ) - { - G_Damage( ent, NULL, NULL, NULL, NULL, 5, DAMAGE_NO_ARMOR, MOD_SUICIDE ); - } - - // lose some voice enthusiasm - if( client->voiceEnthusiasm > 0.0f ) - client->voiceEnthusiasm -= VOICE_ENTHUSIASM_DECAY; - else - client->voiceEnthusiasm = 0.0f; - - client->pers.aliveSeconds++; - if( g_freeFundPeriod.integer > 0 && - client->pers.aliveSeconds % g_freeFundPeriod.integer == 0 ) - { - // Give clients some credit periodically - if( G_TimeTilSuddenDeath( ) > 0 ) - { - if( client->ps.stats[ STAT_TEAM ] == TEAM_ALIENS ) - G_AddCreditToClient( client, FREEKILL_ALIEN, qtrue ); - else if( client->ps.stats[ STAT_TEAM ] == TEAM_HUMANS ) - G_AddCreditToClient( client, FREEKILL_HUMAN, qtrue ); - } - } - } - - while( client->time10000 >= 10000 ) - { - client->time10000 -= 10000; - - if( ent->client->ps.weapon == WP_ABUILD || - ent->client->ps.weapon == WP_ABUILD2 ) - { - AddScore( ent, ALIEN_BUILDER_SCOREINC ); - } - else if( ent->client->ps.weapon == WP_HBUILD ) - { - AddScore( ent, HUMAN_BUILDER_SCOREINC ); - } - - // Give score to basis that healed other aliens - if( ent->client->pers.hasHealed ) - { - if( client->ps.weapon == WP_ALEVEL1 ) - AddScore( ent, LEVEL1_REGEN_SCOREINC ); - else if( client->ps.weapon == WP_ALEVEL1_UPG ) - AddScore( ent, LEVEL1_UPG_REGEN_SCOREINC ); - - ent->client->pers.hasHealed = qfalse; - } - } - - // Regenerate Adv. Dragoon barbs - if( client->ps.weapon == WP_ALEVEL3_UPG ) - { - if( client->ps.ammo < BG_Weapon( WP_ALEVEL3_UPG )->maxAmmo ) - { - if( ent->timestamp + LEVEL3_BOUNCEBALL_REGEN < level.time ) - { - client->ps.ammo++; - ent->timestamp = level.time; - } - } - else - ent->timestamp = level.time; - } -} - -/* -==================== -ClientIntermissionThink -==================== -*/ -void ClientIntermissionThink( gclient_t *client ) -{ - client->ps.eFlags &= ~EF_FIRING; - client->ps.eFlags &= ~EF_FIRING2; - - // the level will exit when everyone wants to or after timeouts - - // swap and latch button actions - client->oldbuttons = client->buttons; - client->buttons = client->pers.cmd.buttons; - if( client->buttons & ( BUTTON_ATTACK | BUTTON_USE_HOLDABLE ) & ( client->oldbuttons ^ client->buttons ) ) - client->readyToExit = 1; -} - - -/* -================ -ClientEvents - -Events will be passed on to the clients for presentation, -but any server game effects are handled here -================ -*/ -void ClientEvents( gentity_t *ent, int oldEventSequence ) -{ - int i; - int event; - gclient_t *client; - int damage; - vec3_t dir; - vec3_t point, mins; - float fallDistance; - class_t class; - - client = ent->client; - class = client->ps.stats[ STAT_CLASS ]; - - if( oldEventSequence < client->ps.eventSequence - MAX_PS_EVENTS ) - oldEventSequence = client->ps.eventSequence - MAX_PS_EVENTS; - - for( i = oldEventSequence; i < client->ps.eventSequence; i++ ) - { - event = client->ps.events[ i & ( MAX_PS_EVENTS - 1 ) ]; - - switch( event ) - { - case EV_FALL_MEDIUM: - case EV_FALL_FAR: - if( ent->s.eType != ET_PLAYER ) - break; // not in the player model - - fallDistance = ( (float)client->ps.stats[ STAT_FALLDIST ] - MIN_FALL_DISTANCE ) / - ( MAX_FALL_DISTANCE - MIN_FALL_DISTANCE ); - - if( fallDistance < 0.0f ) - fallDistance = 0.0f; - else if( fallDistance > 1.0f ) - fallDistance = 1.0f; - - damage = (int)( (float)BG_Class( class )->health * - BG_Class( class )->fallDamage * fallDistance ); - - VectorSet( dir, 0, 0, 1 ); - BG_ClassBoundingBox( class, mins, NULL, NULL, NULL, NULL ); - mins[ 0 ] = mins[ 1 ] = 0.0f; - VectorAdd( client->ps.origin, mins, point ); - - ent->pain_debounce_time = level.time + 200; // no normal pain sound - G_Damage( ent, NULL, NULL, dir, point, damage, DAMAGE_NO_LOCDAMAGE, MOD_FALLING ); - break; - - case EV_FIRE_WEAPON: - FireWeapon( ent ); - break; - - case EV_FIRE_WEAPON2: - FireWeapon2( ent ); - break; - - case EV_FIRE_WEAPON3: - FireWeapon3( ent ); - break; - - case EV_NOAMMO: - break; - - default: - break; - } - } -} - - -/* -============== -SendPendingPredictableEvents -============== -*/ -void SendPendingPredictableEvents( playerState_t *ps ) -{ - gentity_t *t; - int event, seq; - int extEvent, number; - - // if there are still events pending - if( ps->entityEventSequence < ps->eventSequence ) - { - // create a temporary entity for this event which is sent to everyone - // except the client who generated the event - seq = ps->entityEventSequence & ( MAX_PS_EVENTS - 1 ); - event = ps->events[ seq ] | ( ( ps->entityEventSequence & 3 ) << 8 ); - // set external event to zero before calling BG_PlayerStateToEntityState - extEvent = ps->externalEvent; - ps->externalEvent = 0; - // create temporary entity for event - t = G_TempEntity( ps->origin, event ); - number = t->s.number; - BG_PlayerStateToEntityState( ps, &t->s, qtrue ); - t->s.number = number; - t->s.eType = ET_EVENTS + event; - t->s.eFlags |= EF_PLAYER_EVENT; - t->s.otherEntityNum = ps->clientNum; - // send to everyone except the client who generated the event - t->r.svFlags |= SVF_NOTSINGLECLIENT; - t->r.singleClient = ps->clientNum; - // set back external event - ps->externalEvent = extEvent; - } -} - -/* -============== - G_UnlaggedStore - - Called on every server frame. Stores position data for the client at that - into client->unlaggedHist[] and the time into level.unlaggedTimes[]. - This data is used by G_UnlaggedCalc() -============== -*/ -void G_UnlaggedStore( void ) -{ - int i = 0; - gentity_t *ent; - unlagged_t *save; - - if( !g_unlagged.integer ) - return; - level.unlaggedIndex++; - if( level.unlaggedIndex >= MAX_UNLAGGED_MARKERS ) - level.unlaggedIndex = 0; - - level.unlaggedTimes[ level.unlaggedIndex ] = level.time; - - for( i = 0; i < level.maxclients; i++ ) - { - ent = &g_entities[ i ]; - save = &ent->client->unlaggedHist[ level.unlaggedIndex ]; - save->used = qfalse; - if( !ent->r.linked || !( ent->r.contents & CONTENTS_BODY ) ) - continue; - if( ent->client->pers.connected != CON_CONNECTED ) - continue; - VectorCopy( ent->r.mins, save->mins ); - VectorCopy( ent->r.maxs, save->maxs ); - VectorCopy( ent->s.pos.trBase, save->origin ); - save->used = qtrue; - } -} - -/* -============== - G_UnlaggedClear - - Mark all unlaggedHist[] markers for this client invalid. Useful for - preventing teleporting and death. -============== -*/ -void G_UnlaggedClear( gentity_t *ent ) -{ - int i; - - for( i = 0; i < MAX_UNLAGGED_MARKERS; i++ ) - ent->client->unlaggedHist[ i ].used = qfalse; -} - -/* -============== - G_UnlaggedCalc - - Loops through all active clients and calculates their predicted position - for time then stores it in client->unlaggedCalc -============== -*/ -void G_UnlaggedCalc( int time, gentity_t *rewindEnt ) -{ - int i = 0; - gentity_t *ent; - int startIndex = level.unlaggedIndex; - int stopIndex = -1; - int frameMsec = 0; - float lerp = 0.5f; - - if( !g_unlagged.integer ) - return; - - // clear any calculated values from a previous run - for( i = 0; i < level.maxclients; i++ ) - { - ent = &g_entities[ i ]; - ent->client->unlaggedCalc.used = qfalse; - } - - for( i = 0; i < MAX_UNLAGGED_MARKERS; i++ ) - { - if( level.unlaggedTimes[ startIndex ] <= time ) - break; - stopIndex = startIndex; - if( --startIndex < 0 ) - startIndex = MAX_UNLAGGED_MARKERS - 1; - } - if( i == MAX_UNLAGGED_MARKERS ) - { - // if we searched all markers and the oldest one still isn't old enough - // just use the oldest marker with no lerping - lerp = 0.0f; - } - - // client is on the current frame, no need for unlagged - if( stopIndex == -1 ) - return; - - // lerp between two markers - frameMsec = level.unlaggedTimes[ stopIndex ] - - level.unlaggedTimes[ startIndex ]; - if( frameMsec > 0 ) - { - lerp = ( float )( time - level.unlaggedTimes[ startIndex ] ) / - ( float )frameMsec; - } - - for( i = 0; i < level.maxclients; i++ ) - { - ent = &g_entities[ i ]; - if( ent == rewindEnt ) - continue; - if( !ent->r.linked || !( ent->r.contents & CONTENTS_BODY ) ) - continue; - if( ent->client->pers.connected != CON_CONNECTED ) - continue; - if( !ent->client->unlaggedHist[ startIndex ].used ) - continue; - if( !ent->client->unlaggedHist[ stopIndex ].used ) - continue; - - // between two unlagged markers - VectorLerp( lerp, ent->client->unlaggedHist[ startIndex ].mins, - ent->client->unlaggedHist[ stopIndex ].mins, - ent->client->unlaggedCalc.mins ); - VectorLerp( lerp, ent->client->unlaggedHist[ startIndex ].maxs, - ent->client->unlaggedHist[ stopIndex ].maxs, - ent->client->unlaggedCalc.maxs ); - VectorLerp( lerp, ent->client->unlaggedHist[ startIndex ].origin, - ent->client->unlaggedHist[ stopIndex ].origin, - ent->client->unlaggedCalc.origin ); - - ent->client->unlaggedCalc.used = qtrue; - } -} - -/* -============== - G_UnlaggedOff - - Reverses the changes made to all active clients by G_UnlaggedOn() -============== -*/ -void G_UnlaggedOff( void ) -{ - int i = 0; - gentity_t *ent; - - if( !g_unlagged.integer ) - return; - - for( i = 0; i < level.maxclients; i++ ) - { - ent = &g_entities[ i ]; - if( !ent->client->unlaggedBackup.used ) - continue; - VectorCopy( ent->client->unlaggedBackup.mins, ent->r.mins ); - VectorCopy( ent->client->unlaggedBackup.maxs, ent->r.maxs ); - VectorCopy( ent->client->unlaggedBackup.origin, ent->r.currentOrigin ); - ent->client->unlaggedBackup.used = qfalse; - trap_LinkEntity( ent ); - } -} - -/* -============== - G_UnlaggedOn - - Called after G_UnlaggedCalc() to apply the calculated values to all active - clients. Once finished tracing, G_UnlaggedOff() must be called to restore - the clients' position data - - As an optimization, all clients that have an unlagged position that is - not touchable at "range" from "muzzle" will be ignored. This is required - to prevent a huge amount of trap_LinkEntity() calls per user cmd. -============== -*/ - -void G_UnlaggedOn( gentity_t *attacker, vec3_t muzzle, float range ) -{ - int i = 0; - gentity_t *ent; - unlagged_t *calc; - - if( !g_unlagged.integer ) - return; - - if( !attacker->client->pers.useUnlagged ) - return; - - for( i = 0; i < level.maxclients; i++ ) - { - ent = &g_entities[ i ]; - calc = &ent->client->unlaggedCalc; - - if( !calc->used ) - continue; - if( ent->client->unlaggedBackup.used ) - continue; - if( !ent->r.linked || !( ent->r.contents & CONTENTS_BODY ) ) - continue; - if( VectorCompare( ent->r.currentOrigin, calc->origin ) ) - continue; - if( muzzle ) - { - float r1 = Distance( calc->origin, calc->maxs ); - float r2 = Distance( calc->origin, calc->mins ); - float maxRadius = ( r1 > r2 ) ? r1 : r2; - - if( Distance( muzzle, calc->origin ) > range + maxRadius ) - continue; - } - - // create a backup of the real positions - VectorCopy( ent->r.mins, ent->client->unlaggedBackup.mins ); - VectorCopy( ent->r.maxs, ent->client->unlaggedBackup.maxs ); - VectorCopy( ent->r.currentOrigin, ent->client->unlaggedBackup.origin ); - ent->client->unlaggedBackup.used = qtrue; - - // move the client to the calculated unlagged position - VectorCopy( calc->mins, ent->r.mins ); - VectorCopy( calc->maxs, ent->r.maxs ); - VectorCopy( calc->origin, ent->r.currentOrigin ); - trap_LinkEntity( ent ); - } -} -/* -============== - G_UnlaggedDetectCollisions - - cgame prediction will predict a client's own position all the way up to - the current time, but only updates other player's positions up to the - postition sent in the most recent snapshot. - - This allows player X to essentially "move through" the position of player Y - when player X's cmd is processed with Pmove() on the server. This is because - player Y was clipping player X's Pmove() on his client, but when the same - cmd is processed with Pmove on the server it is not clipped. - - Long story short (too late): don't use unlagged positions for players who - were blocking this player X's client-side Pmove(). This makes the assumption - that if player X's movement was blocked in the client he's going to still - be up against player Y when the Pmove() is run on the server with the - same cmd. - - NOTE: this must be called after Pmove() and G_UnlaggedCalc() -============== -*/ -static void G_UnlaggedDetectCollisions( gentity_t *ent ) -{ - unlagged_t *calc; - trace_t tr; - float r1, r2; - float range; - - if( !g_unlagged.integer ) - return; - - if( !ent->client->pers.useUnlagged ) - return; - - calc = &ent->client->unlaggedCalc; - - // if the client isn't moving, this is not necessary - if( VectorCompare( ent->client->oldOrigin, ent->client->ps.origin ) ) - return; - - range = Distance( ent->client->oldOrigin, ent->client->ps.origin ); - - // increase the range by the player's largest possible radius since it's - // the players bounding box that collides, not their origin - r1 = Distance( calc->origin, calc->mins ); - r2 = Distance( calc->origin, calc->maxs ); - range += ( r1 > r2 ) ? r1 : r2; - - G_UnlaggedOn( ent, ent->client->oldOrigin, range ); - - trap_Trace(&tr, ent->client->oldOrigin, ent->r.mins, ent->r.maxs, - ent->client->ps.origin, ent->s.number, MASK_PLAYERSOLID ); - if( tr.entityNum >= 0 && tr.entityNum < MAX_CLIENTS ) - g_entities[ tr.entityNum ].client->unlaggedCalc.used = qfalse; - - G_UnlaggedOff( ); -} - -/* -============== -ClientThink - -This will be called once for each client frame, which will -usually be a couple times for each server frame on fast clients. - -If "g_synchronousClients 1" is set, this will be called exactly -once for each server frame, which makes for smooth demo recording. -============== -*/ -void ClientThink_real( gentity_t *ent ) -{ - gclient_t *client; - pmove_t pm; - int oldEventSequence; - int msec; - usercmd_t *ucmd; - - client = ent->client; - - // don't think if the client is not yet connected (and thus not yet spawned in) - if( client->pers.connected != CON_CONNECTED ) - return; - - // mark the time, so the connection sprite can be removed - ucmd = &ent->client->pers.cmd; - - // sanity check the command time to prevent speedup cheating - if( ucmd->serverTime > level.time + 200 ) - { - ucmd->serverTime = level.time + 200; -// G_Printf("serverTime <<<<<\n" ); - } - - if( ucmd->serverTime < level.time - 1000 ) - { - ucmd->serverTime = level.time - 1000; -// G_Printf("serverTime >>>>>\n" ); - } - - msec = ucmd->serverTime - client->ps.commandTime; - // following others may result in bad times, but we still want - // to check for follow toggles - if( msec < 1 && client->sess.spectatorState != SPECTATOR_FOLLOW ) - return; - - if( msec > 200 ) - msec = 200; - - client->unlaggedTime = ucmd->serverTime; - - if( pmove_msec.integer < 8 ) - trap_Cvar_Set( "pmove_msec", "8" ); - else if( pmove_msec.integer > 33 ) - trap_Cvar_Set( "pmove_msec", "33" ); - - if( pmove_fixed.integer || client->pers.pmoveFixed ) - { - ucmd->serverTime = ( ( ucmd->serverTime + pmove_msec.integer - 1 ) / pmove_msec.integer ) * pmove_msec.integer; - //if (ucmd->serverTime - client->ps.commandTime <= 0) - // return; - } - - // - // check for exiting intermission - // - if( level.intermissiontime ) - { - ClientIntermissionThink( client ); - return; - } - - // spectators don't do much - if( client->sess.spectatorState != SPECTATOR_NOT ) - { - if( client->sess.spectatorState == SPECTATOR_SCOREBOARD ) - return; - - SpectatorThink( ent, ucmd ); - return; - } - - G_namelog_update_score( client ); - - // check for inactivity timer, but never drop the local client of a non-dedicated server - if( !ClientInactivityTimer( ent ) ) - return; - - // calculate where ent is currently seeing all the other active clients - G_UnlaggedCalc( ent->client->unlaggedTime, ent ); - - if( client->noclip ) - client->ps.pm_type = PM_NOCLIP; - else if( client->ps.stats[ STAT_HEALTH ] <= 0 ) - client->ps.pm_type = PM_DEAD; - else if( client->ps.stats[ STAT_STATE ] & SS_BLOBLOCKED || - client->ps.stats[ STAT_STATE ] & SS_GRABBED ) - client->ps.pm_type = PM_GRABBED; - else if( BG_InventoryContainsUpgrade( UP_JETPACK, client->ps.stats ) && BG_UpgradeIsActive( UP_JETPACK, client->ps.stats ) ) - client->ps.pm_type = PM_JETPACK; - else - client->ps.pm_type = PM_NORMAL; - - if( ( client->ps.stats[ STAT_STATE ] & SS_GRABBED ) && - client->grabExpiryTime < level.time ) - client->ps.stats[ STAT_STATE ] &= ~SS_GRABBED; - - if( ( client->ps.stats[ STAT_STATE ] & SS_BLOBLOCKED ) && - client->lastLockTime + LOCKBLOB_LOCKTIME < level.time ) - client->ps.stats[ STAT_STATE ] &= ~SS_BLOBLOCKED; - - if( ( client->ps.stats[ STAT_STATE ] & SS_SLOWLOCKED ) && - client->lastSlowTime + ABUILDER_BLOB_TIME < level.time ) - client->ps.stats[ STAT_STATE ] &= ~SS_SLOWLOCKED; - - // Update boosted state flags - client->ps.stats[ STAT_STATE ] &= ~SS_BOOSTEDWARNING; - if( client->ps.stats[ STAT_STATE ] & SS_BOOSTED ) - { - if( level.time - client->boostedTime >= BOOST_TIME ) - client->ps.stats[ STAT_STATE ] &= ~SS_BOOSTED; - else if( level.time - client->boostedTime >= BOOST_WARN_TIME ) - client->ps.stats[ STAT_STATE ] |= SS_BOOSTEDWARNING; - } - - // Check if poison cloud has worn off - if( ( client->ps.eFlags & EF_POISONCLOUDED ) && - BG_PlayerPoisonCloudTime( &client->ps ) - level.time + - client->lastPoisonCloudedTime <= 0 ) - client->ps.eFlags &= ~EF_POISONCLOUDED; - - if( client->ps.stats[ STAT_STATE ] & SS_POISONED && - client->lastPoisonTime + ALIEN_POISON_TIME < level.time ) - client->ps.stats[ STAT_STATE ] &= ~SS_POISONED; - - client->ps.gravity = g_gravity.value; - - if( BG_InventoryContainsUpgrade( UP_MEDKIT, client->ps.stats ) && - BG_UpgradeIsActive( UP_MEDKIT, client->ps.stats ) ) - { - //if currently using a medkit or have no need for a medkit now - if( client->ps.stats[ STAT_STATE ] & SS_HEALING_2X || - ( client->ps.stats[ STAT_HEALTH ] == client->ps.stats[ STAT_MAX_HEALTH ] && - !( client->ps.stats[ STAT_STATE ] & SS_POISONED ) ) ) - { - BG_DeactivateUpgrade( UP_MEDKIT, client->ps.stats ); - } - else if( client->ps.stats[ STAT_HEALTH ] > 0 ) - { - //remove anti toxin - BG_DeactivateUpgrade( UP_MEDKIT, client->ps.stats ); - BG_RemoveUpgradeFromInventory( UP_MEDKIT, client->ps.stats ); - - client->ps.stats[ STAT_STATE ] &= ~SS_POISONED; - client->poisonImmunityTime = level.time + MEDKIT_POISON_IMMUNITY_TIME; - - client->ps.stats[ STAT_STATE ] |= SS_HEALING_2X; - client->lastMedKitTime = level.time; - client->medKitHealthToRestore = - client->ps.stats[ STAT_MAX_HEALTH ] - client->ps.stats[ STAT_HEALTH ]; - client->medKitIncrementTime = level.time + - ( MEDKIT_STARTUP_TIME / MEDKIT_STARTUP_SPEED ); - - G_AddEvent( ent, EV_MEDKIT_USED, 0 ); - } - } - - // Replenish alien health - if( level.surrenderTeam != client->pers.teamSelection && - ent->nextRegenTime >= 0 && ent->nextRegenTime < level.time ) - { - float regenRate = - BG_Class( ent->client->ps.stats[ STAT_CLASS ] )->regenRate; - - if( ent->health <= 0 || ent->nextRegenTime < 0 || regenRate == 0 ) - ent->nextRegenTime = -1; // no regen - else - { - int entityList[ MAX_GENTITIES ]; - int i, num; - int count, interval; - vec3_t range, mins, maxs; - float modifier = 1.0f; - - VectorSet( range, REGEN_BOOST_RANGE, REGEN_BOOST_RANGE, - REGEN_BOOST_RANGE ); - VectorAdd( client->ps.origin, range, maxs ); - VectorSubtract( client->ps.origin, range, mins ); - - num = trap_EntitiesInBox( mins, maxs, entityList, MAX_GENTITIES ); - for( i = 0; i < num; i++ ) - { - gentity_t *boost = &g_entities[ entityList[ i ] ]; - - if( Distance( client->ps.origin, boost->s.origin ) > REGEN_BOOST_RANGE ) - continue; - - if( modifier < BOOSTER_REGEN_MOD && boost->s.eType == ET_BUILDABLE && - boost->s.modelindex == BA_A_BOOSTER && boost->spawned && - boost->health > 0 && boost->powered ) - { - modifier = BOOSTER_REGEN_MOD; - continue; - } - - if( boost->s.eType == ET_PLAYER && boost->client && - boost->client->pers.teamSelection == - ent->client->pers.teamSelection && boost->health > 0 ) - { - class_t class = boost->client->ps.stats[ STAT_CLASS ]; - qboolean didBoost = qfalse; - - if( class == PCL_ALIEN_LEVEL1 && modifier < LEVEL1_REGEN_MOD ) - { - modifier = LEVEL1_REGEN_MOD; - didBoost = qtrue; - } - else if( class == PCL_ALIEN_LEVEL1_UPG && - modifier < LEVEL1_UPG_REGEN_MOD ) - { - modifier = LEVEL1_UPG_REGEN_MOD; - didBoost = qtrue; - } - - if( didBoost && ent->health < client->ps.stats[ STAT_MAX_HEALTH ] ) - boost->client->pers.hasHealed = qtrue; - } - } - - // Transmit heal rate to the client so it can be displayed on the HUD - client->ps.stats[ STAT_STATE ] |= SS_HEALING_ACTIVE; - client->ps.stats[ STAT_STATE ] &= ~( SS_HEALING_2X | SS_HEALING_3X ); - if( modifier == 1.0f && !G_FindCreep( ent ) ) - { - client->ps.stats[ STAT_STATE ] &= ~SS_HEALING_ACTIVE; - modifier *= ALIEN_REGEN_NOCREEP_MOD; - } - else if( modifier >= 3.0f ) - client->ps.stats[ STAT_STATE ] |= SS_HEALING_3X; - else if( modifier >= 2.0f ) - client->ps.stats[ STAT_STATE ] |= SS_HEALING_2X; - - interval = 1000 / ( regenRate * modifier ); - // if recovery interval is less than frametime, compensate - count = 1 + ( level.time - ent->nextRegenTime ) / interval; - - ent->health += count; - ent->nextRegenTime += count * interval; - - // if at max health, clear damage counters - if( ent->health >= client->ps.stats[ STAT_MAX_HEALTH ] ) - { - ent->health = client->ps.stats[ STAT_MAX_HEALTH ]; - for( i = 0; i < MAX_CLIENTS; i++ ) - ent->credits[ i ] = 0; - } - } - } - - if( BG_InventoryContainsUpgrade( UP_GRENADE, client->ps.stats ) && - BG_UpgradeIsActive( UP_GRENADE, client->ps.stats ) ) - { - int lastWeapon = ent->s.weapon; - - //remove grenade - BG_DeactivateUpgrade( UP_GRENADE, client->ps.stats ); - BG_RemoveUpgradeFromInventory( UP_GRENADE, client->ps.stats ); - - //M-M-M-M-MONSTER HACK - ent->s.weapon = WP_GRENADE; - FireWeapon( ent ); - ent->s.weapon = lastWeapon; - } - - // set speed - if( client->ps.pm_type == PM_NOCLIP ) - client->ps.speed = client->pers.flySpeed; - else - client->ps.speed = g_speed.value * - BG_Class( client->ps.stats[ STAT_CLASS ] )->speed; - - if( client->lastCreepSlowTime + CREEP_TIMEOUT < level.time ) - client->ps.stats[ STAT_STATE ] &= ~SS_CREEPSLOWED; - - //randomly disable the jet pack if damaged - if( BG_InventoryContainsUpgrade( UP_JETPACK, client->ps.stats ) && - BG_UpgradeIsActive( UP_JETPACK, client->ps.stats ) ) - { - if( ent->lastDamageTime + JETPACK_DISABLE_TIME > level.time ) - { - if( random( ) > JETPACK_DISABLE_CHANCE ) - client->ps.pm_type = PM_NORMAL; - } - - //switch jetpack off if no reactor - //if( !G_Reactor( ) ) - // BG_DeactivateUpgrade( UP_JETPACK, client->ps.stats ); - } - - // set up for pmove - oldEventSequence = client->ps.eventSequence; - - memset( &pm, 0, sizeof( pm ) ); - - if( ent->flags & FL_FORCE_GESTURE ) - { - ent->flags &= ~FL_FORCE_GESTURE; - ent->client->pers.cmd.buttons |= BUTTON_GESTURE; - } - - // clear fall velocity before every pmove - client->pmext.fallVelocity = 0.0f; - - pm.ps = &client->ps; - pm.pmext = &client->pmext; - pm.cmd = *ucmd; - - if( pm.ps->pm_type == PM_DEAD ) - pm.tracemask = MASK_DEADSOLID; - else - pm.tracemask = MASK_PLAYERSOLID; - - pm.trace = G_PMTraceHack; - pm.pointcontents = trap_PointContents; - pm.debugLevel = g_debugMove.integer; - pm.noFootsteps = 0; - - pm.pmove_fixed = pmove_fixed.integer | client->pers.pmoveFixed; - pm.pmove_msec = pmove_msec.integer; - - VectorCopy( client->ps.origin, client->oldOrigin ); - - // moved from after Pmove -- potentially the cause of - // future triggering bugs - if( !ent->client->noclip ) - G_TouchTriggers( ent ); - - Pmove( &pm ); - - G_UnlaggedDetectCollisions( ent ); - - // save results of pmove - if( ent->client->ps.eventSequence != oldEventSequence ) - ent->eventTime = level.time; - - if( g_smoothClients.integer ) - BG_PlayerStateToEntityStateExtraPolate( &ent->client->ps, &ent->s, ent->client->ps.commandTime, qtrue ); - else - BG_PlayerStateToEntityState( &ent->client->ps, &ent->s, qtrue ); - - switch( client->ps.weapon ) - { - case WP_ALEVEL0: - if( !CheckVenomAttack( ent ) ) - { - client->ps.weaponstate = WEAPON_READY; - } - else - { - client->ps.generic1 = WPM_PRIMARY; - G_AddEvent( ent, EV_FIRE_WEAPON, 0 ); - } - break; - - case WP_ALEVEL1: - case WP_ALEVEL1_UPG: - CheckGrabAttack( ent ); - break; - - case WP_ALEVEL3: - case WP_ALEVEL3_UPG: - if( !CheckPounceAttack( ent ) ) - { - client->ps.weaponstate = WEAPON_READY; - } - else - { - client->ps.generic1 = WPM_SECONDARY; - G_AddEvent( ent, EV_FIRE_WEAPON2, 0 ); - } - break; - - case WP_ALEVEL4: - // If not currently in a trample, reset the trample bookkeeping data - if( !( client->ps.pm_flags & PMF_CHARGE ) && client->trampleBuildablesHitPos ) - { - ent->client->trampleBuildablesHitPos = 0; - memset( ent->client->trampleBuildablesHit, 0, sizeof( ent->client->trampleBuildablesHit ) ); - } - break; - - case WP_HBUILD: - CheckCkitRepair( ent ); - break; - - default: - break; - } - - SendPendingPredictableEvents( &ent->client->ps ); - - if( !( ent->client->ps.eFlags & EF_FIRING ) ) - client->fireHeld = qfalse; // for grapple - if( !( ent->client->ps.eFlags & EF_FIRING2 ) ) - client->fire2Held = qfalse; - - // use the snapped origin for linking so it matches client predicted versions - VectorCopy( ent->s.pos.trBase, ent->r.currentOrigin ); - - VectorCopy( pm.mins, ent->r.mins ); - VectorCopy( pm.maxs, ent->r.maxs ); - - ent->waterlevel = pm.waterlevel; - ent->watertype = pm.watertype; - - // touch other objects - ClientImpacts( ent, &pm ); - - // execute client events - ClientEvents( ent, oldEventSequence ); - - // link entity now, after any personal teleporters have been used - trap_LinkEntity( ent ); - - // NOTE: now copy the exact origin over otherwise clients can be snapped into solid - VectorCopy( ent->client->ps.origin, ent->r.currentOrigin ); - VectorCopy( ent->client->ps.origin, ent->s.origin ); - - // save results of triggers and client events - if( ent->client->ps.eventSequence != oldEventSequence ) - ent->eventTime = level.time; - - // Don't think anymore if dead - if( client->ps.stats[ STAT_HEALTH ] <= 0 ) - return; - - // swap and latch button actions - client->oldbuttons = client->buttons; - client->buttons = ucmd->buttons; - client->latched_buttons |= client->buttons & ~client->oldbuttons; - - if( ( client->buttons & BUTTON_USE_EVOLVE ) && !( client->oldbuttons & BUTTON_USE_EVOLVE ) && - client->ps.stats[ STAT_HEALTH ] > 0 ) - { - trace_t trace; - vec3_t eyes, view, point; - gentity_t *traceEnt; - -#define USE_OBJECT_RANGE 64 - - int entityList[ MAX_GENTITIES ]; - vec3_t range = { USE_OBJECT_RANGE, USE_OBJECT_RANGE, USE_OBJECT_RANGE }; - vec3_t mins, maxs; - int i, num; - - // look for object infront of player - AngleVectors( client->ps.viewangles, view, NULL, NULL ); - BG_GetClientViewOrigin( &client->ps, eyes ); // !@#CUBOID - VectorMA( eyes, USE_OBJECT_RANGE, view, point ); - - trap_Trace( &trace, client->ps.origin, NULL, NULL, point, ent->s.number, MASK_SHOT ); - traceEnt = &g_entities[ trace.entityNum ]; - - if( traceEnt && traceEnt->buildableTeam == client->ps.stats[ STAT_TEAM ] && traceEnt->use ) - traceEnt->use( traceEnt, ent, ent ); //other and activator are the same in this context - else - { - //no entity in front of player - do a small area search - - VectorAdd( client->ps.origin, range, maxs ); - VectorSubtract( client->ps.origin, range, mins ); - - num = trap_EntitiesInBox( mins, maxs, entityList, MAX_GENTITIES ); - for( i = 0; i < num; i++ ) - { - traceEnt = &g_entities[ entityList[ i ] ]; - - if( traceEnt && traceEnt->buildableTeam == client->ps.stats[ STAT_TEAM ] && traceEnt->use ) - { - traceEnt->use( traceEnt, ent, ent ); //other and activator are the same in this context - break; - } - } - - if( i == num && client->ps.stats[ STAT_TEAM ] == TEAM_ALIENS ) - { - if( BG_AlienCanEvolve( client->ps.stats[ STAT_CLASS ], - client->pers.credit, - g_alienStage.integer ) ) - { - //no nearby objects and alien - show class menu - G_TriggerMenu( ent->client->ps.clientNum, MN_A_INFEST ); - } - else - { - //flash frags - G_AddEvent( ent, EV_ALIEN_EVOLVE_FAILED, 0 ); - } - } - } - } - - client->ps.persistant[ PERS_BP ] = G_GetBuildPoints( client->ps.origin, - client->ps.stats[ STAT_TEAM ] ); - client->ps.persistant[ PERS_MARKEDBP ] = G_GetMarkedBuildPoints( client->ps.origin, - client->ps.stats[ STAT_TEAM ] ); - - if( client->ps.persistant[ PERS_BP ] < 0 ) - client->ps.persistant[ PERS_BP ] = 0; - - // perform once-a-second actions - ClientTimerActions( ent, msec ); - - if( ent->suicideTime > 0 && ent->suicideTime < level.time ) - { - ent->flags &= ~FL_GODMODE; - ent->client->ps.stats[ STAT_HEALTH ] = ent->health = 0; - player_die( ent, ent, ent, 100000, MOD_SUICIDE ); - - ent->suicideTime = 0; - } -} - -/* -================== -ClientThink - -A new command has arrived from the client -================== -*/ -void ClientThink( int clientNum ) -{ - gentity_t *ent; - - ent = g_entities + clientNum; - trap_GetUsercmd( clientNum, &ent->client->pers.cmd ); - - // mark the time we got info, so we can display the - // phone jack if they don't get any for a while - ent->client->lastCmdTime = level.time; - - if( !g_synchronousClients.integer ) - ClientThink_real( ent ); -} - - -void G_RunClient( gentity_t *ent ) -{ - if( !g_synchronousClients.integer ) - return; - - ent->client->pers.cmd.serverTime = level.time; - ClientThink_real( ent ); -} - - -/* -================== -SpectatorClientEndFrame - -================== -*/ -void SpectatorClientEndFrame( gentity_t *ent ) -{ - gclient_t *cl; - int clientNum; - int score, ping; - - // if we are doing a chase cam or a remote view, grab the latest info - if( ent->client->sess.spectatorState == SPECTATOR_FOLLOW ) - { - clientNum = ent->client->sess.spectatorClient; - if( clientNum >= 0 && clientNum < level.maxclients ) - { - cl = &level.clients[ clientNum ]; - if( cl->pers.connected == CON_CONNECTED ) - { - score = ent->client->ps.persistant[ PERS_SCORE ]; - ping = ent->client->ps.ping; - - // Copy - ent->client->ps = cl->ps; - - // Restore - ent->client->ps.persistant[ PERS_SCORE ] = score; - ent->client->ps.ping = ping; - - ent->client->ps.pm_flags |= PMF_FOLLOW; - ent->client->ps.pm_flags &= ~PMF_QUEUED; - } - } - } -} - -/* -============== -ClientEndFrame - -Called at the end of each server frame for each connected client -A fast client will have multiple ClientThink for each ClientEdFrame, -while a slow client may have multiple ClientEndFrame between ClientThink. -============== -*/ -void ClientEndFrame( gentity_t *ent ) -{ - clientPersistant_t *pers; - - if( ent->client->sess.spectatorState != SPECTATOR_NOT ) - { - SpectatorClientEndFrame( ent ); - return; - } - - pers = &ent->client->pers; - - // - // If the end of unit layout is displayed, don't give - // the player any normal movement attributes - // - if( level.intermissiontime ) - return; - - // burn from lava, etc - P_WorldEffects( ent ); - - // apply all the damage taken this frame - P_DamageFeedback( ent ); - - // add the EF_CONNECTION flag if we haven't gotten commands recently - if( level.time - ent->client->lastCmdTime > 1000 ) - ent->s.eFlags |= EF_CONNECTION; - else - ent->s.eFlags &= ~EF_CONNECTION; - - ent->client->ps.stats[ STAT_HEALTH ] = ent->health; // FIXME: get rid of ent->health... - - // respawn if dead - if( ent->client->ps.stats[ STAT_HEALTH ] <= 0 && level.time >= ent->client->respawnTime ) - respawn( ent ); - - G_SetClientSound( ent ); - - // set the latest infor - if( g_smoothClients.integer ) - BG_PlayerStateToEntityStateExtraPolate( &ent->client->ps, &ent->s, ent->client->ps.commandTime, qtrue ); - else - BG_PlayerStateToEntityState( &ent->client->ps, &ent->s, qtrue ); - - SendPendingPredictableEvents( &ent->client->ps ); -} - diff --git a/src/game/g_client.c.orig b/src/game/g_client.c.orig deleted file mode 100644 index 735c59d..0000000 --- a/src/game/g_client.c.orig +++ /dev/null @@ -1,1600 +0,0 @@ -/* -=========================================================================== -Copyright (C) 1999-2005 Id Software, Inc. -Copyright (C) 2000-2009 Darklegion Development - -This file is part of Tremulous. - -Tremulous 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. - -Tremulous is distributed in the hope that it will be -useful, but WITHOUT ANY WARRANTY; without even the implied warranty of -MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -GNU General Public License for more details. - -You should have received a copy of the GNU General Public License -along with Tremulous; if not, write to the Free Software -Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA -=========================================================================== -*/ - -#include "g_local.h" - -// g_client.c -- client functions that don't happen every frame - -static vec3_t playerMins = {-15, -15, -24}; -static vec3_t playerMaxs = {15, 15, 32}; - -/*QUAKED info_player_deathmatch (1 0 1) (-16 -16 -24) (16 16 32) initial -potential spawning position for deathmatch games. -The first time a player enters the game, they will be at an 'initial' spot. -Targets will be fired when someone spawns in on them. -"nobots" will prevent bots from using this spot. -"nohumans" will prevent non-bots from using this spot. -*/ -void SP_info_player_deathmatch( gentity_t *ent ) -{ - int i; - - G_SpawnInt( "nobots", "0", &i); - - if( i ) - ent->flags |= FL_NO_BOTS; - - G_SpawnInt( "nohumans", "0", &i ); - if( i ) - ent->flags |= FL_NO_HUMANS; -} - -/*QUAKED info_player_start (1 0 0) (-16 -16 -24) (16 16 32) -equivelant to info_player_deathmatch -*/ -void SP_info_player_start( gentity_t *ent ) -{ - ent->classname = "info_player_deathmatch"; - SP_info_player_deathmatch( ent ); -} - -/*QUAKED info_player_intermission (1 0 1) (-16 -16 -24) (16 16 32) -The intermission will be viewed from this point. Target an info_notnull for the view direction. -*/ -void SP_info_player_intermission( gentity_t *ent ) -{ -} - -/*QUAKED info_alien_intermission (1 0 1) (-16 -16 -24) (16 16 32) -The intermission will be viewed from this point. Target an info_notnull for the view direction. -*/ -void SP_info_alien_intermission( gentity_t *ent ) -{ -} - -/*QUAKED info_human_intermission (1 0 1) (-16 -16 -24) (16 16 32) -The intermission will be viewed from this point. Target an info_notnull for the view direction. -*/ -void SP_info_human_intermission( gentity_t *ent ) -{ -} - -/* -=============== -G_AddCreditToClient -=============== -*/ -void G_AddCreditToClient( gclient_t *client, short credit, qboolean cap ) -{ - int capAmount; - - if( !client ) - return; - - if( cap && credit > 0 ) - { - capAmount = client->pers.teamSelection == TEAM_ALIENS ? - ALIEN_MAX_CREDITS : HUMAN_MAX_CREDITS; - if( client->pers.credit < capAmount ) - { - client->pers.credit += credit; - if( client->pers.credit > capAmount ) - client->pers.credit = capAmount; - } - } - else - client->pers.credit += credit; - - if( client->pers.credit < 0 ) - client->pers.credit = 0; - - // Copy to ps so the client can access it - client->ps.persistant[ PERS_CREDIT ] = client->pers.credit; -} - - -/* -======================================================================= - - G_SelectSpawnPoint - -======================================================================= -*/ - -/* -================ -SpotWouldTelefrag - -================ -*/ -qboolean SpotWouldTelefrag( gentity_t *spot ) -{ - int i, num; - int touch[ MAX_GENTITIES ]; - gentity_t *hit; - vec3_t mins, maxs; - - VectorAdd( spot->s.origin, playerMins, mins ); - VectorAdd( spot->s.origin, playerMaxs, maxs ); - num = trap_EntitiesInBox( mins, maxs, touch, MAX_GENTITIES ); - - for( i = 0; i < num; i++ ) - { - hit = &g_entities[ touch[ i ] ]; - //if ( hit->client && hit->client->ps.stats[STAT_HEALTH] > 0 ) { - if( hit->client ) - return qtrue; - } - - return qfalse; -} - - -/* -=========== -G_SelectRandomFurthestSpawnPoint - -Chooses a player start, deathmatch start, etc -============ -*/ -static gentity_t *G_SelectRandomFurthestSpawnPoint ( vec3_t avoidPoint, vec3_t origin, vec3_t angles ) -{ - gentity_t *spot; - vec3_t delta; - float dist; - float list_dist[ 64 ]; - gentity_t *list_spot[ 64 ]; - int numSpots, rnd, i, j; - - numSpots = 0; - spot = NULL; - - while( ( spot = G_Find( spot, FOFS( classname ), "info_player_deathmatch" ) ) != NULL ) - { - if( SpotWouldTelefrag( spot ) ) - continue; - - VectorSubtract( spot->s.origin, avoidPoint, delta ); - dist = VectorLength( delta ); - - for( i = 0; i < numSpots; i++ ) - { - if( dist > list_dist[ i ] ) - { - if( numSpots >= 64 ) - numSpots = 64 - 1; - - for( j = numSpots; j > i; j-- ) - { - list_dist[ j ] = list_dist[ j - 1 ]; - list_spot[ j ] = list_spot[ j - 1 ]; - } - - list_dist[ i ] = dist; - list_spot[ i ] = spot; - numSpots++; - - if( numSpots > 64 ) - numSpots = 64; - - break; - } - } - - if( i >= numSpots && numSpots < 64 ) - { - list_dist[ numSpots ] = dist; - list_spot[ numSpots ] = spot; - numSpots++; - } - } - - if( !numSpots ) - { - spot = G_Find( NULL, FOFS( classname ), "info_player_deathmatch" ); - - if( !spot ) - G_Error( "Couldn't find a spawn point" ); - - VectorCopy( spot->s.origin, origin ); - origin[ 2 ] += 9; - VectorCopy( spot->s.angles, angles ); - return spot; - } - - // select a random spot from the spawn points furthest away - rnd = random( ) * ( numSpots / 2 ); - - VectorCopy( list_spot[ rnd ]->s.origin, origin ); - origin[ 2 ] += 9; - VectorCopy( list_spot[ rnd ]->s.angles, angles ); - - return list_spot[ rnd ]; -} - - -/* -================ -G_SelectSpawnBuildable - -find the nearest buildable of the right type that is -spawned/healthy/unblocked etc. -================ -*/ -static gentity_t *G_SelectSpawnBuildable( vec3_t preference, buildable_t buildable ) -{ - gentity_t *search, *spot; - - search = spot = NULL; - - while( ( search = G_Find( search, FOFS( classname ), - BG_Buildable( buildable, NULL )->entityName ) ) != NULL ) - { - if( !search->spawned ) - continue; - - if( search->health <= 0 ) - continue; - - if( !search->s.groundEntityNum ) - continue; - - if( search->clientSpawnTime > 0 ) - continue; - - if( G_CheckSpawnPoint( search->s.number, search->s.origin, - search->s.origin2, buildable, NULL ) != NULL ) - continue; - - if( !spot || DistanceSquared( preference, search->s.origin ) < - DistanceSquared( preference, spot->s.origin ) ) - spot = search; - } - - return spot; -} - -/* -=========== -G_SelectSpawnPoint - -Chooses a player start, deathmatch start, etc -============ -*/ -gentity_t *G_SelectSpawnPoint( vec3_t avoidPoint, vec3_t origin, vec3_t angles ) -{ - return G_SelectRandomFurthestSpawnPoint( avoidPoint, origin, angles ); -} - - -/* -=========== -G_SelectTremulousSpawnPoint - -Chooses a player start, deathmatch start, etc -============ -*/ -gentity_t *G_SelectTremulousSpawnPoint( team_t team, vec3_t preference, vec3_t origin, vec3_t angles ) -{ - gentity_t *spot = NULL; - - if( team == TEAM_ALIENS ) - { - if( level.numAlienSpawns <= 0 ) - return NULL; - - spot = G_SelectSpawnBuildable( preference, BA_A_SPAWN ); - } - else if( team == TEAM_HUMANS ) - { - if( level.numHumanSpawns <= 0 ) - return NULL; - - spot = G_SelectSpawnBuildable( preference, BA_H_SPAWN ); - } - - //no available spots - if( !spot ) - return NULL; - - if( team == TEAM_ALIENS ) - G_CheckSpawnPoint( spot->s.number, spot->s.origin, spot->s.origin2, BA_A_SPAWN, origin ); - else if( team == TEAM_HUMANS ) - G_CheckSpawnPoint( spot->s.number, spot->s.origin, spot->s.origin2, BA_H_SPAWN, origin ); - - VectorCopy( spot->s.angles, angles ); - angles[ ROLL ] = 0; - - return spot; - -} - - -/* -=========== -G_SelectSpectatorSpawnPoint - -============ -*/ -static gentity_t *G_SelectSpectatorSpawnPoint( vec3_t origin, vec3_t angles ) -{ - FindIntermissionPoint( ); - - VectorCopy( level.intermission_origin, origin ); - VectorCopy( level.intermission_angle, angles ); - - return NULL; -} - - -/* -=========== -G_SelectAlienLockSpawnPoint - -Try to find a spawn point for alien intermission otherwise -use normal intermission spawn. -============ -*/ -gentity_t *G_SelectAlienLockSpawnPoint( vec3_t origin, vec3_t angles ) -{ - gentity_t *spot; - - spot = NULL; - spot = G_Find( spot, FOFS( classname ), "info_alien_intermission" ); - - if( !spot ) - return G_SelectSpectatorSpawnPoint( origin, angles ); - - VectorCopy( spot->s.origin, origin ); - VectorCopy( spot->s.angles, angles ); - - return spot; -} - - -/* -=========== -G_SelectHumanLockSpawnPoint - -Try to find a spawn point for human intermission otherwise -use normal intermission spawn. -============ -*/ -gentity_t *G_SelectHumanLockSpawnPoint( vec3_t origin, vec3_t angles ) -{ - gentity_t *spot; - - spot = NULL; - spot = G_Find( spot, FOFS( classname ), "info_human_intermission" ); - - if( !spot ) - return G_SelectSpectatorSpawnPoint( origin, angles ); - - VectorCopy( spot->s.origin, origin ); - VectorCopy( spot->s.angles, angles ); - - return spot; -} - - -/* -======================================================================= - -BODYQUE - -======================================================================= -*/ - - -/* -============= -BodySink - -After sitting around for five seconds, fall into the ground and dissapear -============= -*/ -static void BodySink( gentity_t *ent ) -{ - //run on first BodySink call - if( !ent->active ) - { - ent->active = qtrue; - - //sinking bodies can't be infested - ent->killedBy = ent->s.misc = MAX_CLIENTS; - ent->timestamp = level.time; - } - - if( level.time - ent->timestamp > 6500 ) - { - G_FreeEntity( ent ); - return; - } - - ent->nextthink = level.time + 100; - ent->s.pos.trBase[ 2 ] -= 1; -} - - -/* -============= -SpawnCorpse - -A player is respawning, so make an entity that looks -just like the existing corpse to leave behind. -============= -*/ -static void SpawnCorpse( gentity_t *ent ) -{ - gentity_t *body; - int contents; - vec3_t origin, dest; - trace_t tr; - float vDiff; - - VectorCopy( ent->r.currentOrigin, origin ); - - trap_UnlinkEntity( ent ); - - // if client is in a nodrop area, don't leave the body - contents = trap_PointContents( origin, -1 ); - if( contents & CONTENTS_NODROP ) - return; - - body = G_Spawn( ); - - VectorCopy( ent->s.apos.trBase, body->s.angles ); - body->s.eFlags = EF_DEAD; - body->s.eType = ET_CORPSE; - body->s.number = body - g_entities; - body->timestamp = level.time; - body->s.event = 0; - body->r.contents = CONTENTS_CORPSE; - body->s.clientNum = ent->client->ps.stats[ STAT_CLASS ]; - body->nonSegModel = ent->client->ps.persistant[ PERS_STATE ] & PS_NONSEGMODEL; - - if( ent->client->ps.stats[ STAT_TEAM ] == TEAM_HUMANS ) - body->classname = "humanCorpse"; - else - body->classname = "alienCorpse"; - - body->s.misc = MAX_CLIENTS; - - body->think = BodySink; - body->nextthink = level.time + 20000; - - body->s.legsAnim = ent->s.legsAnim; - - if( !body->nonSegModel ) - { - switch( body->s.legsAnim & ~ANIM_TOGGLEBIT ) - { - case BOTH_DEATH1: - case BOTH_DEAD1: - body->s.torsoAnim = body->s.legsAnim = BOTH_DEAD1; - break; - case BOTH_DEATH2: - case BOTH_DEAD2: - body->s.torsoAnim = body->s.legsAnim = BOTH_DEAD2; - break; - case BOTH_DEATH3: - case BOTH_DEAD3: - default: - body->s.torsoAnim = body->s.legsAnim = BOTH_DEAD3; - break; - } - } - else - { - switch( body->s.legsAnim & ~ANIM_TOGGLEBIT ) - { - case NSPA_DEATH1: - case NSPA_DEAD1: - body->s.legsAnim = NSPA_DEAD1; - break; - case NSPA_DEATH2: - case NSPA_DEAD2: - body->s.legsAnim = NSPA_DEAD2; - break; - case NSPA_DEATH3: - case NSPA_DEAD3: - default: - body->s.legsAnim = NSPA_DEAD3; - break; - } - } - - body->takedamage = qfalse; - - body->health = ent->health = ent->client->ps.stats[ STAT_HEALTH ]; - ent->health = 0; - - //change body dimensions - BG_ClassBoundingBox( ent->client->ps.stats[ STAT_CLASS ], NULL, NULL, NULL, body->r.mins, body->r.maxs ); - vDiff = body->r.mins[ 2 ] - ent->r.mins[ 2 ]; - - //drop down to match the *model* origins of ent and body - VectorSet( dest, origin[ 0 ], origin[ 1 ], origin[ 2 ] - vDiff ); - trap_Trace( &tr, origin, body->r.mins, body->r.maxs, dest, body->s.number, body->clipmask ); - VectorCopy( tr.endpos, origin ); - - G_SetOrigin( body, origin ); - VectorCopy( origin, body->s.origin ); - body->s.pos.trType = TR_GRAVITY; - body->s.pos.trTime = level.time; - VectorCopy( ent->client->ps.velocity, body->s.pos.trDelta ); - - VectorCopy ( body->s.pos.trBase, body->r.currentOrigin ); - trap_LinkEntity( body ); -} - -//====================================================================== - - -/* -================== -G_SetClientViewAngle - -================== -*/ -void G_SetClientViewAngle( gentity_t *ent, vec3_t angle ) -{ - int i; - - // set the delta angle - for( i = 0; i < 3; i++ ) - { - int cmdAngle; - - cmdAngle = ANGLE2SHORT( angle[ i ] ); - ent->client->ps.delta_angles[ i ] = cmdAngle - ent->client->pers.cmd.angles[ i ]; - } - - VectorCopy( angle, ent->s.angles ); - VectorCopy( ent->s.angles, ent->client->ps.viewangles ); -} - -/* -================ -respawn -================ -*/ -void respawn( gentity_t *ent ) -{ - int i; - - SpawnCorpse( ent ); - - // Clients can't respawn - they must go through the class cmd - ent->client->pers.classSelection = PCL_NONE; - ClientSpawn( ent, NULL, NULL, NULL ); - - // stop any following clients that don't have sticky spec on - for( i = 0; i < level.maxclients; i++ ) - { - if( level.clients[ i ].sess.spectatorState == SPECTATOR_FOLLOW && - level.clients[ i ].sess.spectatorClient == ent - g_entities ) - { - if( !( level.clients[ i ].pers.stickySpec ) ) - { - if( !G_FollowNewClient( &g_entities[ i ], 1 ) ) - G_StopFollowing( &g_entities[ i ] ); - } - else - G_FollowLockView( &g_entities[ i ] ); - } - } -} - -static qboolean G_IsEmoticon( const char *s, qboolean *escaped ) -{ - int i, j; - const char *p = s; - char emoticon[ MAX_EMOTICON_NAME_LEN ] = {""}; - qboolean escape = qfalse; - - if( *p != '[' ) - return qfalse; - p++; - if( *p == '[' ) - { - escape = qtrue; - p++; - } - i = 0; - while( *p && i < ( MAX_EMOTICON_NAME_LEN - 1 ) ) - { - if( *p == ']' ) - { - for( j = 0; j < level.emoticonCount; j++ ) - { - if( !Q_stricmp( emoticon, level.emoticons[ j ].name ) ) - { - *escaped = escape; - return qtrue; - } - } - return qfalse; - } - emoticon[ i++ ] = *p; - emoticon[ i ] = '\0'; - p++; - } - return qfalse; -} - -/* -=========== -G_ClientCleanName -============ -*/ -static void G_ClientCleanName( const char *in, char *out, int outSize ) -{ - int len, colorlessLen; - char *p; - int spaces; - qboolean escaped; - qboolean invalid = qfalse; - - //save room for trailing null byte - outSize--; - - len = 0; - colorlessLen = 0; - p = out; - *p = 0; - spaces = 0; - - for( ; *in; in++ ) - { - // don't allow leading spaces - if( colorlessLen == 0 && *in == ' ' ) - continue; - - // don't allow nonprinting characters or (dead) console keys - if( *in < ' ' || *in > '}' || *in == '`' ) - continue; - - // check colors - if( Q_IsColorString( in ) ) - { - in++; - - // make sure room in dest for both chars - if( len > outSize - 2 ) - break; - - *out++ = Q_COLOR_ESCAPE; - - *out++ = *in; - - len += 2; - continue; - } - else if( !g_emoticonsAllowedInNames.integer && G_IsEmoticon( in, &escaped ) ) - { - // make sure room in dest for both chars - if( len > outSize - 2 ) - break; - - *out++ = '['; - *out++ = '['; - len += 2; - if( escaped ) - in++; - continue; - } - - // don't allow too many consecutive spaces - if( *in == ' ' ) - { - spaces++; - if( spaces > 3 ) - continue; - } - else - spaces = 0; - - if( len > outSize - 1 ) - break; - - *out++ = *in; - colorlessLen++; - len++; - } - - *out = 0; - - // don't allow names beginning with "[skipnotify]" because it messes up /ignore-related code - if( !Q_stricmpn( p, "[skipnotify]", 12 ) ) - invalid = qtrue; - - // don't allow comment-beginning strings because it messes up various parsers - if( strstr( p, "//" ) || strstr( p, "/*" ) ) - invalid = qtrue; - - // don't allow empty names - if( *p == 0 || colorlessLen == 0 ) - invalid = qtrue; - - // if something made the name bad, put them back to UnnamedPlayer - if( invalid ) - Q_strncpyz( p, "UnnamedPlayer", outSize ); -} - - -/* -====================== -G_NonSegModel - -Reads an animation.cfg to check for nonsegmentation -====================== -*/ -static qboolean G_NonSegModel( const char *filename ) -{ - char *text_p; - int len; - char *token; - char text[ 20000 ]; - fileHandle_t f; - - // load the file - len = trap_FS_FOpenFile( filename, &f, FS_READ ); - if( !f ) - { - G_Printf( "File not found: %s\n", filename ); - return qfalse; - } - - if( len < 0 ) - return qfalse; - - if( len == 0 || len >= sizeof( text ) - 1 ) - { - trap_FS_FCloseFile( f ); - G_Printf( "File %s is %s\n", filename, len == 0 ? "empty" : "too long" ); - return qfalse; - } - - trap_FS_Read( text, len, f ); - text[ len ] = 0; - trap_FS_FCloseFile( f ); - - // parse the text - text_p = text; - - // read optional parameters - while( 1 ) - { - token = COM_Parse( &text_p ); - - //EOF - if( !token[ 0 ] ) - break; - - if( !Q_stricmp( token, "nonsegmented" ) ) - return qtrue; - } - - return qfalse; -} - -/* -=========== -ClientUserInfoChanged - -Called from ClientConnect when the player first connects and -directly by the server system when the player updates a userinfo variable. - -The game can override any of the settings and call trap_SetUserinfo -if desired. -============ -*/ -char *ClientUserinfoChanged( int clientNum, qboolean forceName ) -{ - gentity_t *ent; - char *s; - char model[ MAX_QPATH ]; - char buffer[ MAX_QPATH ]; - char filename[ MAX_QPATH ]; - char oldname[ MAX_NAME_LENGTH ]; - char newname[ MAX_NAME_LENGTH ]; - char err[ MAX_STRING_CHARS ]; - qboolean revertName = qfalse; - gclient_t *client; - char userinfo[ MAX_INFO_STRING ]; - - ent = g_entities + clientNum; - client = ent->client; - - trap_GetUserinfo( clientNum, userinfo, sizeof( userinfo ) ); - - // check for malformed or illegal info strings - if( !Info_Validate(userinfo) ) - { - trap_SendServerCommand( ent - g_entities, - "disconnect \"illegal or malformed userinfo\n\"" ); - trap_DropClient( ent - g_entities, - "dropped: illegal or malformed userinfo"); - return "Illegal or malformed userinfo"; - } - // If their userinfo overflowed, tremded is in the process of disconnecting them. - // If we send our own disconnect, it won't work, so just return to prevent crashes later - // in this function. This check must come after the Info_Validate call. - else if( !userinfo[ 0 ] ) - return "Empty (overflowed) userinfo"; - - // stickyspec toggle - s = Info_ValueForKey( userinfo, "cg_stickySpec" ); - client->pers.stickySpec = atoi( s ) != 0; - - // set name - Q_strncpyz( oldname, client->pers.netname, sizeof( oldname ) ); - s = Info_ValueForKey( userinfo, "name" ); - G_ClientCleanName( s, newname, sizeof( newname ) ); - - if( strcmp( oldname, newname ) ) - { - if( !forceName && client->pers.namelog->nameChangeTime && - level.time - client->pers.namelog->nameChangeTime <= - g_minNameChangePeriod.value * 1000 ) - { - trap_SendServerCommand( ent - g_entities, va( - "print \"Name change spam protection (g_minNameChangePeriod = %d)\n\"", - g_minNameChangePeriod.integer ) ); - revertName = qtrue; - } - else if( !forceName && g_maxNameChanges.integer > 0 && - client->pers.namelog->nameChanges >= g_maxNameChanges.integer ) - { - trap_SendServerCommand( ent - g_entities, va( - "print \"Maximum name changes reached (g_maxNameChanges = %d)\n\"", - g_maxNameChanges.integer ) ); - revertName = qtrue; - } - else if( !forceName && client->pers.namelog->muted ) - { - trap_SendServerCommand( ent - g_entities, - "print \"You cannot change your name while you are muted\n\"" ); - revertName = qtrue; - } - else if( !G_admin_name_check( ent, newname, err, sizeof( err ) ) ) - { - trap_SendServerCommand( ent - g_entities, va( "print \"%s\n\"", err ) ); - revertName = qtrue; - } - - if( revertName ) - { - Q_strncpyz( client->pers.netname, *oldname ? oldname : "UnnamedPlayer", - sizeof( client->pers.netname ) ); - Info_SetValueForKey( userinfo, "name", oldname ); - trap_SetUserinfo( clientNum, userinfo ); - } - else - { - G_CensorString( client->pers.netname, newname, - sizeof( client->pers.netname ), ent ); - if( !forceName && client->pers.connected == CON_CONNECTED ) - { - client->pers.namelog->nameChangeTime = level.time; - client->pers.namelog->nameChanges++; - } - if( *oldname ) - { - G_LogPrintf( "ClientRename: %i [%s] (%s) \"%s^7\" -> \"%s^7\" \"%c%s%c^7\"\n", - clientNum, client->pers.ip.str, client->pers.guid, - oldname, client->pers.netname, - DECOLOR_OFF, client->pers.netname, DECOLOR_ON ); - } - } - G_namelog_update_name( client ); - } - - if( client->pers.classSelection == PCL_NONE ) - { - //This looks hacky and frankly it is. The clientInfo string needs to hold different - //model details to that of the spawning class or the info change will not be - //registered and an axis appears instead of the player model. There is zero chance - //the player can spawn with the battlesuit, hence this choice. - Com_sprintf( buffer, MAX_QPATH, "%s/%s", BG_ClassConfig( PCL_HUMAN_BSUIT )->modelName, - BG_ClassConfig( PCL_HUMAN_BSUIT )->skinName ); - } - else - { - Com_sprintf( buffer, MAX_QPATH, "%s/%s", BG_ClassConfig( client->pers.classSelection )->modelName, - BG_ClassConfig( client->pers.classSelection )->skinName ); - - //model segmentation - Com_sprintf( filename, sizeof( filename ), "models/players/%s/animation.cfg", - BG_ClassConfig( client->pers.classSelection )->modelName ); - - if( G_NonSegModel( filename ) ) - client->ps.persistant[ PERS_STATE ] |= PS_NONSEGMODEL; - else - client->ps.persistant[ PERS_STATE ] &= ~PS_NONSEGMODEL; - } - Q_strncpyz( model, buffer, sizeof( model ) ); - - // wallwalk follow - s = Info_ValueForKey( userinfo, "cg_wwFollow" ); - - if( atoi( s ) ) - client->ps.persistant[ PERS_STATE ] |= PS_WALLCLIMBINGFOLLOW; - else - client->ps.persistant[ PERS_STATE ] &= ~PS_WALLCLIMBINGFOLLOW; - - // wallwalk toggle - s = Info_ValueForKey( userinfo, "cg_wwToggle" ); - - if( atoi( s ) ) - client->ps.persistant[ PERS_STATE ] |= PS_WALLCLIMBINGTOGGLE; - else - client->ps.persistant[ PERS_STATE ] &= ~PS_WALLCLIMBINGTOGGLE; - - // always sprint - s = Info_ValueForKey( userinfo, "cg_sprintToggle" ); - - if( atoi( s ) ) - client->ps.persistant[ PERS_STATE ] |= PS_SPRINTTOGGLE; - else - client->ps.persistant[ PERS_STATE ] &= ~PS_SPRINTTOGGLE; - - // fly speed - s = Info_ValueForKey( userinfo, "cg_flySpeed" ); - - if( *s ) - client->pers.flySpeed = atoi( s ); - else - client->pers.flySpeed = BG_Class( PCL_NONE )->speed; - - // disable blueprint errors - s = Info_ValueForKey( userinfo, "cg_disableBlueprintErrors" ); - - if( atoi( s ) ) - client->pers.disableBlueprintErrors = qtrue; - else - client->pers.disableBlueprintErrors = qfalse; - - // teamInfo - s = Info_ValueForKey( userinfo, "teamoverlay" ); - - if( atoi( s ) != 0 ) - client->pers.teamInfo = qtrue; - else - client->pers.teamInfo = qfalse; - - s = Info_ValueForKey( userinfo, "cg_unlagged" ); - if( !s[0] || atoi( s ) != 0 ) - client->pers.useUnlagged = qtrue; - else - client->pers.useUnlagged = qfalse; - - Q_strncpyz( client->pers.voice, Info_ValueForKey( userinfo, "voice" ), - sizeof( client->pers.voice ) ); - - // send over a subset of the userinfo keys so other clients can - // print scoreboards, display models, and play custom sounds - - Com_sprintf( userinfo, sizeof( userinfo ), - "n\\%s\\t\\%i\\model\\%s\\ig\\%16s\\v\\%s", - client->pers.netname, client->pers.teamSelection, model, - Com_ClientListString( &client->sess.ignoreList ), - client->pers.voice ); - - trap_SetConfigstring( CS_PLAYERS + clientNum, userinfo ); - - /*G_LogPrintf( "ClientUserinfoChanged: %i %s\n", clientNum, userinfo );*/ - - return NULL; -} - - -/* -=========== -ClientConnect - -Called when a player begins connecting to the server. -Called again for every map change or tournement restart. - -The session information will be valid after exit. - -Return NULL if the client should be allowed, otherwise return -a string with the reason for denial. - -Otherwise, the client will be sent the current gamestate -and will eventually get to ClientBegin. - -firstTime will be qtrue the very first time a client connects -to the server machine, but qfalse on map changes and tournement -restarts. -============ -*/ -char *ClientConnect( int clientNum, qboolean firstTime ) -{ - char *value; - char *userInfoError; - gclient_t *client; - char userinfo[ MAX_INFO_STRING ]; - gentity_t *ent; - char reason[ MAX_STRING_CHARS ] = {""}; - int i; - - ent = &g_entities[ clientNum ]; - client = &level.clients[ clientNum ]; - - // ignore if client already connected - if( client->pers.connected != CON_DISCONNECTED ) - return NULL; - - ent->client = client; - memset( client, 0, sizeof( *client ) ); - - trap_GetUserinfo( clientNum, userinfo, sizeof( userinfo ) ); - - value = Info_ValueForKey( userinfo, "cl_guid" ); - Q_strncpyz( client->pers.guid, value, sizeof( client->pers.guid ) ); - - value = Info_ValueForKey( userinfo, "ip" ); - // check for local client - if( !strcmp( value, "localhost" ) ) - client->pers.localClient = qtrue; - G_AddressParse( value, &client->pers.ip ); - - client->pers.admin = G_admin_admin( client->pers.guid ); - - // check for admin ban - if( G_admin_ban_check( ent, reason, sizeof( reason ) ) ) - { - return va( "%s", reason ); - } - - // check for a password - value = Info_ValueForKey( userinfo, "password" ); - - if( g_password.string[ 0 ] && Q_stricmp( g_password.string, "none" ) && - strcmp( g_password.string, value ) != 0 ) - return "Invalid password"; - - // add guid to session so we don't have to keep parsing userinfo everywhere - for( i = 0; i < sizeof( client->pers.guid ) - 1 && - isxdigit( client->pers.guid[ i ] ); i++ ); - - if( i < sizeof( client->pers.guid ) - 1 ) - return "Invalid GUID"; - - for( i = 0; i < level.maxclients; i++ ) - { - if( level.clients[ i ].pers.connected == CON_DISCONNECTED ) - continue; - - if( !Q_stricmp( client->pers.guid, level.clients[ i ].pers.guid ) ) - { - if( !G_ClientIsLagging( level.clients + i ) ) - { - trap_SendServerCommand( i, "cp \"Your GUID is not secure\"" ); - return "Duplicate GUID"; - } - trap_DropClient( i, "Ghost" ); - } - } - - client->pers.connected = CON_CONNECTING; - - // read or initialize the session data - if( firstTime || level.newSession ) - G_InitSessionData( client, userinfo ); - - G_ReadSessionData( client ); - - // get and distribute relevent paramters - G_namelog_connect( client ); - userInfoError = ClientUserinfoChanged( clientNum, qfalse ); - if( userInfoError != NULL ) - return userInfoError; - - G_LogPrintf( "ClientConnect: %i [%s] (%s) \"%s^7\" \"%c%s%c^7\"\n", - clientNum, client->pers.ip.str, client->pers.guid, - client->pers.netname, - DECOLOR_OFF, client->pers.netname, DECOLOR_ON ); - - // don't do the "xxx connected" messages if they were caried over from previous level - if( firstTime ) - trap_SendServerCommand( -1, va( "print \"%s" S_COLOR_WHITE " connected\n\"", - client->pers.netname ) ); - - if( client->pers.admin ) - G_admin_authlog( ent ); - - // count current clients and rank for scoreboard - CalculateRanks( ); - - - // if this is after !restart keepteams or !restart switchteams, apply said selection - if ( client->sess.restartTeam != TEAM_NONE ) - { - G_ChangeTeam( ent, client->sess.restartTeam ); - client->sess.restartTeam = TEAM_NONE; - } - - - return NULL; -} - -/* -=========== -ClientBegin - -called when a client has finished connecting, and is ready -to be placed into the level. This will happen every level load, -and on transition between teams, but doesn't happen on respawns -============ -*/ -void ClientBegin( int clientNum ) -{ - gentity_t *ent; - gclient_t *client; - int flags; - - ent = g_entities + clientNum; - - client = level.clients + clientNum; - - // ignore if client already entered the game - if( client->pers.connected != CON_CONNECTING ) - return; - - if( ent->r.linked ) - trap_UnlinkEntity( ent ); - - G_InitGentity( ent ); - ent->touch = 0; - ent->pain = 0; - ent->client = client; - - client->pers.connected = CON_CONNECTED; - client->pers.enterTime = level.time; - - // save eflags around this, because changing teams will - // cause this to happen with a valid entity, and we - // want to make sure the teleport bit is set right - // so the viewpoint doesn't interpolate through the - // world to the new position - flags = client->ps.eFlags; - memset( &client->ps, 0, sizeof( client->ps ) ); - memset( &client->pmext, 0, sizeof( client->pmext ) ); - client->ps.eFlags = flags; - - // locate ent at a spawn point - ClientSpawn( ent, NULL, NULL, NULL ); - - trap_SendServerCommand( -1, va( "print \"%s" S_COLOR_WHITE " entered the game\n\"", client->pers.netname ) ); - - G_namelog_restore( client ); - - G_LogPrintf( "ClientBegin: %i\n", clientNum ); - - // count current clients and rank for scoreboard - CalculateRanks( ); - - // send the client a list of commands that can be used - G_ListCommands( ent ); - - // reset cuboidSelection - client->cuboidSelection[ 0 ] = - client->cuboidSelection[ 1 ] = - client->cuboidSelection[ 2 ] = 32; -} - -/* -=========== -ClientSpawn - -Called every time a client is placed fresh in the world: -after the first ClientBegin, and after each respawn -Initializes all non-persistant parts of playerState -============ -*/ -void ClientSpawn( gentity_t *ent, gentity_t *spawn, vec3_t origin, vec3_t angles ) -{ - int index; - vec3_t spawn_origin, spawn_angles; - gclient_t *client; - int i; - clientPersistant_t saved; - clientSession_t savedSess; - int persistant[ MAX_PERSISTANT ]; - gentity_t *spawnPoint = NULL; - int flags; - int savedPing; - int teamLocal; - int eventSequence; - char userinfo[ MAX_INFO_STRING ]; - vec3_t up = { 0.0f, 0.0f, 1.0f }; - int maxAmmo, maxClips; - weapon_t weapon; - - index = ent - g_entities; - client = ent->client; - - teamLocal = client->pers.teamSelection; - - //if client is dead and following teammate, stop following before spawning - if( client->sess.spectatorClient != -1 ) - { - client->sess.spectatorClient = -1; - client->sess.spectatorState = SPECTATOR_FREE; - } - - // only start client if chosen a class and joined a team - if( client->pers.classSelection == PCL_NONE && teamLocal == TEAM_NONE ) - client->sess.spectatorState = SPECTATOR_FREE; - else if( client->pers.classSelection == PCL_NONE ) - client->sess.spectatorState = SPECTATOR_LOCKED; - - // if client is dead and following teammate, stop following before spawning - if( ent->client->sess.spectatorState == SPECTATOR_FOLLOW ) - G_StopFollowing( ent ); - - if( origin != NULL ) - VectorCopy( origin, spawn_origin ); - - if( angles != NULL ) - VectorCopy( angles, spawn_angles ); - - // find a spawn point - // do it before setting health back up, so farthest - // ranging doesn't count this client - if( client->sess.spectatorState != SPECTATOR_NOT ) - { - if( teamLocal == TEAM_NONE ) - spawnPoint = G_SelectSpectatorSpawnPoint( spawn_origin, spawn_angles ); - else if( teamLocal == TEAM_ALIENS ) - spawnPoint = G_SelectAlienLockSpawnPoint( spawn_origin, spawn_angles ); - else if( teamLocal == TEAM_HUMANS ) - spawnPoint = G_SelectHumanLockSpawnPoint( spawn_origin, spawn_angles ); - } - else - { - if( spawn == NULL ) - { - G_Error( "ClientSpawn: spawn is NULL\n" ); - return; - } - - spawnPoint = spawn; - - if( ent != spawn ) - { - //start spawn animation on spawnPoint - G_SetBuildableAnim( spawnPoint, BANIM_SPAWN1, qtrue ); - - if( spawnPoint->buildableTeam == TEAM_ALIENS ) - spawnPoint->clientSpawnTime = ALIEN_SPAWN_REPEAT_TIME; - else if( spawnPoint->buildableTeam == TEAM_HUMANS ) - spawnPoint->clientSpawnTime = HUMAN_SPAWN_REPEAT_TIME; - } - } - - // toggle the teleport bit so the client knows to not lerp - flags = ( ent->client->ps.eFlags & EF_TELEPORT_BIT ) ^ EF_TELEPORT_BIT; - G_UnlaggedClear( ent ); - - // clear everything but the persistant data - - saved = client->pers; - savedSess = client->sess; - savedPing = client->ps.ping; - - for( i = 0; i < MAX_PERSISTANT; i++ ) - persistant[ i ] = client->ps.persistant[ i ]; - - eventSequence = client->ps.eventSequence; - memset( client, 0, sizeof( *client ) ); - - client->pers = saved; - client->sess = savedSess; - client->ps.ping = savedPing; - client->lastkilled_client = -1; - - for( i = 0; i < MAX_PERSISTANT; i++ ) - client->ps.persistant[ i ] = persistant[ i ]; - - client->ps.eventSequence = eventSequence; - - // increment the spawncount so the client will detect the respawn - client->ps.persistant[ PERS_SPAWN_COUNT ]++; - client->ps.persistant[ PERS_SPECSTATE ] = client->sess.spectatorState; - - client->airOutTime = level.time + 12000; - - trap_GetUserinfo( index, userinfo, sizeof( userinfo ) ); - client->ps.eFlags = flags; - - //Com_Printf( "ent->client->pers->pclass = %i\n", ent->client->pers.classSelection ); - - ent->s.groundEntityNum = ENTITYNUM_NONE; - ent->client = &level.clients[ index ]; - ent->takedamage = qtrue; - ent->inuse = qtrue; - ent->classname = "player"; - ent->r.contents = CONTENTS_BODY; - ent->clipmask = MASK_PLAYERSOLID; - ent->die = player_die; - ent->waterlevel = 0; - ent->watertype = 0; - ent->flags = 0; - - // calculate each client's acceleration - ent->evaluateAcceleration = qtrue; - - client->ps.stats[ STAT_MISC ] = 0; - client->buildTimer = 0; - - client->ps.eFlags = flags; - client->ps.clientNum = index; - - BG_ClassBoundingBox( ent->client->pers.classSelection, ent->r.mins, ent->r.maxs, NULL, NULL, NULL ); - - if( client->sess.spectatorState == SPECTATOR_NOT ) - client->ps.stats[ STAT_MAX_HEALTH ] = - BG_Class( ent->client->pers.classSelection )->health; - else - client->ps.stats[ STAT_MAX_HEALTH ] = 100; - - // clear entity values - if( ent->client->pers.classSelection == PCL_HUMAN ) - { - BG_AddUpgradeToInventory( UP_MEDKIT, client->ps.stats ); - weapon = client->pers.humanItemSelection; - } - else if( client->sess.spectatorState == SPECTATOR_NOT ) - weapon = BG_Class( ent->client->pers.classSelection )->startWeapon; - else - weapon = WP_NONE; - - maxAmmo = BG_Weapon( weapon )->maxAmmo; - maxClips = BG_Weapon( weapon )->maxClips; - client->ps.stats[ STAT_WEAPON ] = weapon; - client->ps.ammo = maxAmmo; - client->ps.clips = maxClips; - - // We just spawned, not changing weapons - client->ps.persistant[ PERS_NEWWEAPON ] = 0; - - ent->client->ps.stats[ STAT_CLASS ] = ent->client->pers.classSelection; - ent->client->ps.stats[ STAT_TEAM ] = ent->client->pers.teamSelection; - - ent->client->ps.stats[ STAT_BUILDABLE ] = BA_NONE; - ent->client->ps.stats[ STAT_STATE ] = 0; - VectorSet( ent->client->ps.grapplePoint, 0.0f, 0.0f, 1.0f ); - - // health will count down towards max_health - ent->health = client->ps.stats[ STAT_HEALTH ] = client->ps.stats[ STAT_MAX_HEALTH ]; //* 1.25; - - //if evolving scale health - if( ent == spawn ) - { - ent->health *= ent->client->pers.evolveHealthFraction; - client->ps.stats[ STAT_HEALTH ] *= ent->client->pers.evolveHealthFraction; - } - - //clear the credits array - for( i = 0; i < MAX_CLIENTS; i++ ) - ent->credits[ i ] = 0; - - client->ps.stats[ STAT_STAMINA ] = STAMINA_MAX; - - G_SetOrigin( ent, spawn_origin ); - VectorCopy( spawn_origin, client->ps.origin ); - -#define UP_VEL 150.0f -#define F_VEL 50.0f - - //give aliens some spawn velocity - if( client->sess.spectatorState == SPECTATOR_NOT && - client->ps.stats[ STAT_TEAM ] == TEAM_ALIENS ) - { - if( ent == spawn ) - { - //evolution particle system - G_AddPredictableEvent( ent, EV_ALIEN_EVOLVE, DirToByte( up ) ); - } - else - { - spawn_angles[ YAW ] += 180.0f; - AngleNormalize360( spawn_angles[ YAW ] ); - - if( spawnPoint->s.origin2[ 2 ] > 0.0f ) - { - vec3_t forward, dir; - - AngleVectors( spawn_angles, forward, NULL, NULL ); - VectorScale( forward, F_VEL, forward ); - VectorAdd( spawnPoint->s.origin2, forward, dir ); - VectorNormalize( dir ); - - VectorScale( dir, UP_VEL, client->ps.velocity ); - } - - G_AddPredictableEvent( ent, EV_PLAYER_RESPAWN, 0 ); - } - } - else if( client->sess.spectatorState == SPECTATOR_NOT && - client->ps.stats[ STAT_TEAM ] == TEAM_HUMANS ) - { - spawn_angles[ YAW ] += 180.0f; - AngleNormalize360( spawn_angles[ YAW ] ); - } - - // the respawned flag will be cleared after the attack and jump keys come up - client->ps.pm_flags |= PMF_RESPAWNED; - - trap_GetUsercmd( client - level.clients, &ent->client->pers.cmd ); - G_SetClientViewAngle( ent, spawn_angles ); - - if( client->sess.spectatorState == SPECTATOR_NOT ) - { - trap_LinkEntity( ent ); - - // force the base weapon up - if( client->pers.teamSelection == TEAM_HUMANS ) - G_ForceWeaponChange( ent, weapon ); - - client->ps.weaponstate = WEAPON_READY; - } - - // don't allow full run speed for a bit - client->ps.pm_flags |= PMF_TIME_KNOCKBACK; - client->ps.pm_time = 100; - - client->respawnTime = level.time; - ent->nextRegenTime = level.time; - - client->inactivityTime = level.time + g_inactivity.integer * 1000; - client->latched_buttons = 0; - - // set default animations - client->ps.torsoAnim = TORSO_STAND; - client->ps.legsAnim = LEGS_IDLE; - - if( level.intermissiontime ) - MoveClientToIntermission( ent ); - else - { - // fire the targets of the spawn point - if( !spawn ) - G_UseTargets( spawnPoint, ent ); - - // select the highest weapon number available, after any - // spawn given items have fired - client->ps.weapon = 1; - - for( i = WP_NUM_WEAPONS - 1; i > 0 ; i-- ) - { - if( BG_InventoryContainsWeapon( i, client->ps.stats ) ) - { - client->ps.weapon = i; - break; - } - } - } - - client->lastRantBombTime = level.time; - - // run a client frame to drop exactly to the floor, - // initialize animations and other things - client->ps.commandTime = level.time - 100; - ent->client->pers.cmd.serverTime = level.time; - ClientThink( ent-g_entities ); - - // positively link the client, even if the command times are weird - if( client->sess.spectatorState == SPECTATOR_NOT ) - { - BG_PlayerStateToEntityState( &client->ps, &ent->s, qtrue ); - VectorCopy( ent->client->ps.origin, ent->r.currentOrigin ); - trap_LinkEntity( ent ); - } - - // must do this here so the number of active clients is calculated - CalculateRanks( ); - - // run the presend to set anything else - ClientEndFrame( ent ); - - // clear entity state values - BG_PlayerStateToEntityState( &client->ps, &ent->s, qtrue ); -} - - -/* -=========== -ClientDisconnect - -Called when a player drops from the server. -Will not be called between levels. - -This should NOT be called directly by any game logic, -call trap_DropClient(), which will call this and do -server system housekeeping. -============ -*/ -void ClientDisconnect( int clientNum ) -{ - gentity_t *ent; - gentity_t *tent; - int i; - - ent = g_entities + clientNum; - - if( !ent->client || ent->client->pers.connected == CON_DISCONNECTED ) - return; - - G_LeaveTeam( ent ); - G_namelog_disconnect( ent->client ); - G_Vote( ent, TEAM_NONE, qfalse ); - - // stop any following clients - for( i = 0; i < level.maxclients; i++ ) - { - // remove any /ignore settings for this clientNum - Com_ClientListRemove( &level.clients[ i ].sess.ignoreList, clientNum ); - } - - // send effect if they were completely connected - if( ent->client->pers.connected == CON_CONNECTED && - ent->client->sess.spectatorState == SPECTATOR_NOT ) - { - tent = G_TempEntity( ent->client->ps.origin, EV_PLAYER_TELEPORT_OUT ); - tent->s.clientNum = ent->s.clientNum; - } - - G_LogPrintf( "ClientDisconnect: %i [%s] (%s) \"%s^7\"\n", clientNum, - ent->client->pers.ip.str, ent->client->pers.guid, ent->client->pers.netname ); - - trap_UnlinkEntity( ent ); - ent->s.modelindex = 0; - ent->inuse = qfalse; - ent->classname = "disconnected"; - ent->client->pers.connected = CON_DISCONNECTED; - ent->client->sess.spectatorState = - ent->client->ps.persistant[ PERS_SPECSTATE ] = SPECTATOR_NOT; - - trap_SetConfigstring( CS_PLAYERS + clientNum, ""); - - CalculateRanks( ); -} - -/* -=========== -G_RelayCuboidToSpectators - -Called everytime a player changes his cuboid size. -A server command is issued to everyone spectating him -so that their clients can know the cuboid size as well. -============ -*/ -void G_RelayCuboidToSpectators(gentity_t *self) -{ -} - diff --git a/src/game/g_cmds.c.orig b/src/game/g_cmds.c.orig deleted file mode 100644 index 5a30f7b..0000000 --- a/src/game/g_cmds.c.orig +++ /dev/null @@ -1,3588 +0,0 @@ -/* -=========================================================================== -Copyright (C) 1999-2005 Id Software, Inc. -Copyright (C) 2000-2009 Darklegion Development - -This file is part of Tremulous. - -Tremulous 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. - -Tremulous is distributed in the hope that it will be -useful, but WITHOUT ANY WARRANTY; without even the implied warranty of -MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -GNU General Public License for more details. - -You should have received a copy of the GNU General Public License -along with Tremulous; if not, write to the Free Software -Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA -=========================================================================== -*/ - -#include "g_local.h" - -/* -================== -G_SanitiseString - -Remove color codes and non-alphanumeric characters from a string -================== -*/ -void G_SanitiseString( char *in, char *out, int len ) -{ - len--; - - while( *in && len > 0 ) - { - if( Q_IsColorString( in ) ) - { - in += 2; // skip color code - continue; - } - - if( isalnum( *in ) ) - { - *out++ = tolower( *in ); - len--; - } - in++; - } - *out = 0; -} - -/* -================== -G_ClientNumberFromString - -Returns a player number for either a number or name string -Returns -1 and optionally sets err if invalid or not exactly 1 match -err will have a trailing \n if set -================== -*/ -int G_ClientNumberFromString( char *s, char *err, int len ) -{ - gclient_t *cl; - int i, found = 0, m = -1; - char s2[ MAX_NAME_LENGTH ]; - char n2[ MAX_NAME_LENGTH ]; - char *p = err; - int l, l2 = len; - - if( !s[ 0 ] ) - { - if( p ) - Q_strncpyz( p, "no player name or slot # provided\n", len ); - - return -1; - } - - // numeric values are just slot numbers - for( i = 0; s[ i ] && isdigit( s[ i ] ); i++ ); - if( !s[ i ] ) - { - i = atoi( s ); - - if( i < 0 || i >= level.maxclients ) - return -1; - - cl = &level.clients[ i ]; - - if( cl->pers.connected == CON_DISCONNECTED ) - { - if( p ) - Q_strncpyz( p, "no player connected in that slot #\n", len ); - - return -1; - } - - return i; - } - - G_SanitiseString( s, s2, sizeof( s2 ) ); - if( !s2[ 0 ] ) - { - if( p ) - Q_strncpyz( p, "no player name provided\n", len ); - - return -1; - } - - if( p ) - { - Q_strncpyz( p, "more than one player name matches. " - "be more specific or use the slot #:\n", l2 ); - l = strlen( p ); - p += l; - l2 -= l; - } - - // check for a name match - for( i = 0, cl = level.clients; i < level.maxclients; i++, cl++ ) - { - if( cl->pers.connected == CON_DISCONNECTED ) - continue; - - G_SanitiseString( cl->pers.netname, n2, sizeof( n2 ) ); - - if( !strcmp( n2, s2 ) ) - return i; - - if( strstr( n2, s2 ) ) - { - if( p ) - { - l = Q_snprintf( p, l2, "%-2d - %s^7\n", i, cl->pers.netname ); - p += l; - l2 -= l; - } - - found++; - m = i; - } - } - - if( found == 1 ) - return m; - - if( found == 0 && err ) - Q_strncpyz( err, "no connected player by that name or slot #\n", len ); - - return -1; -} - -/* -================== -G_ClientNumbersFromString - -Sets plist to an array of integers that represent client numbers that have -names that are a partial match for s. - -Returns number of matching clientids up to max. -================== -*/ -int G_ClientNumbersFromString( char *s, int *plist, int max ) -{ - gclient_t *p; - int i, found = 0; - char *endptr; - char n2[ MAX_NAME_LENGTH ] = {""}; - char s2[ MAX_NAME_LENGTH ] = {""}; - - if( max == 0 ) - return 0; - - if( !s[ 0 ] ) - return 0; - - // if a number is provided, it is a clientnum - i = strtol( s, &endptr, 10 ); - if( *endptr == '\0' ) - { - if( i >= 0 && i < level.maxclients ) - { - p = &level.clients[ i ]; - if( p->pers.connected != CON_DISCONNECTED ) - { - *plist = i; - return 1; - } - } - // we must assume that if only a number is provided, it is a clientNum - return 0; - } - - // now look for name matches - G_SanitiseString( s, s2, sizeof( s2 ) ); - if( !s2[ 0 ] ) - return 0; - for( i = 0; i < level.maxclients && found < max; i++ ) - { - p = &level.clients[ i ]; - if( p->pers.connected == CON_DISCONNECTED ) - { - continue; - } - G_SanitiseString( p->pers.netname, n2, sizeof( n2 ) ); - if( strstr( n2, s2 ) ) - { - *plist++ = i; - found++; - } - } - return found; -} - -/* -================== -ScoreboardMessage - -================== -*/ -void ScoreboardMessage( gentity_t *ent ) -{ - char entry[ 1024 ]; - char string[ 1400 ]; - int stringlength; - int i, j; - gclient_t *cl; - int numSorted; - weapon_t weapon = WP_NONE; - upgrade_t upgrade = UP_NONE; - - // send the latest information on all clients - string[ 0 ] = 0; - stringlength = 0; - - numSorted = level.numConnectedClients; - - for( i = 0; i < numSorted; i++ ) - { - int ping; - - cl = &level.clients[ level.sortedClients[ i ] ]; - - if( cl->pers.connected == CON_CONNECTING ) - ping = -1; - else - ping = cl->ps.ping < 999 ? cl->ps.ping : 999; - - if( cl->sess.spectatorState == SPECTATOR_NOT && - ( ent->client->pers.teamSelection == TEAM_NONE || - cl->pers.teamSelection == ent->client->pers.teamSelection ) ) - { - weapon = cl->ps.weapon; - - if( BG_InventoryContainsUpgrade( UP_BATTLESUIT, cl->ps.stats ) ) - upgrade = UP_BATTLESUIT; - else if( BG_InventoryContainsUpgrade( UP_JETPACK, cl->ps.stats ) ) - upgrade = UP_JETPACK; - else if( BG_InventoryContainsUpgrade( UP_BATTPACK, cl->ps.stats ) ) - upgrade = UP_BATTPACK; - else if( BG_InventoryContainsUpgrade( UP_HELMET_MK1, cl->ps.stats ) ) - upgrade = UP_HELMET_MK1; - else if( BG_InventoryContainsUpgrade( UP_HELMET_MK2, cl->ps.stats ) ) - upgrade = UP_HELMET_MK2; - else if( BG_InventoryContainsUpgrade( UP_LIGHTARMOUR, cl->ps.stats ) ) - upgrade = UP_LIGHTARMOUR; - else - upgrade = UP_NONE; - } - else - { - weapon = WP_NONE; - upgrade = UP_NONE; - } - - Com_sprintf( entry, sizeof( entry ), - " %d %d %d %d %d %d", level.sortedClients[ i ], cl->ps.persistant[ PERS_SCORE ], - ping, ( level.time - cl->pers.enterTime ) / 60000, weapon, upgrade ); - - j = strlen( entry ); - - if( stringlength + j >= 1024 ) - break; - - strcpy( string + stringlength, entry ); - stringlength += j; - } - - trap_SendServerCommand( ent-g_entities, va( "scores %i %i%s", - level.alienKills, level.humanKills, string ) ); -} - - -/* -================== -ConcatArgs -================== -*/ -char *ConcatArgs( int start ) -{ - int i, c, tlen; - static char line[ MAX_STRING_CHARS ]; - int len; - char arg[ MAX_STRING_CHARS ]; - - len = 0; - c = trap_Argc( ); - - for( i = start; i < c; i++ ) - { - trap_Argv( i, arg, sizeof( arg ) ); - tlen = strlen( arg ); - - if( len + tlen >= MAX_STRING_CHARS - 1 ) - break; - - memcpy( line + len, arg, tlen ); - len += tlen; - - if( len == MAX_STRING_CHARS - 1 ) - break; - - if( i != c - 1 ) - { - line[ len ] = ' '; - len++; - } - } - - line[ len ] = 0; - - return line; -} - -/* -================== -ConcatArgsPrintable -Duplicate of concatargs but enquotes things that need to be -Used to log command arguments in a way that preserves user intended tokenizing -================== -*/ -char *ConcatArgsPrintable( int start ) -{ - int i, c, tlen; - static char line[ MAX_STRING_CHARS ]; - int len; - char arg[ MAX_STRING_CHARS + 2 ]; - char *printArg; - - len = 0; - c = trap_Argc( ); - - for( i = start; i < c; i++ ) - { - printArg = arg; - trap_Argv( i, arg, sizeof( arg ) ); - if( strchr( arg, ' ' ) ) - printArg = va( "\"%s\"", arg ); - tlen = strlen( printArg ); - - if( len + tlen >= MAX_STRING_CHARS - 1 ) - break; - - memcpy( line + len, printArg, tlen ); - len += tlen; - - if( len == MAX_STRING_CHARS - 1 ) - break; - - if( i != c - 1 ) - { - line[ len ] = ' '; - len++; - } - } - - line[ len ] = 0; - - return line; -} - - -/* -================== -Cmd_Give_f - -Give items to a client -================== -*/ -void Cmd_Give_f( gentity_t *ent ) -{ - char *name; - qboolean give_all = qfalse; - - if( trap_Argc( ) < 2 ) - { - ADMP( "usage: give [what]\n" ); - ADMP( "usage: valid choices are: all, health, funds [amount], stamina, " - "poison, gas, ammo\n" ); - return; - } - - name = ConcatArgs( 1 ); - if( Q_stricmp( name, "all" ) == 0 ) - give_all = qtrue; - - if( give_all || Q_stricmp( name, "health" ) == 0 ) - { - ent->health = ent->client->ps.stats[ STAT_MAX_HEALTH ]; - BG_AddUpgradeToInventory( UP_MEDKIT, ent->client->ps.stats ); - } - - if( give_all || Q_stricmpn( name, "funds", 5 ) == 0 ) - { - float credits; - - if( give_all || trap_Argc( ) < 3 ) - credits = 30000.0f; - else - { - credits = atof( name + 6 ) * - ( ent->client->pers.teamSelection == - TEAM_ALIENS ? ALIEN_CREDITS_PER_KILL : 1.0f ); - - // clamp credits manually, as G_AddCreditToClient() expects a short int - if( credits > SHRT_MAX ) - credits = 30000.0f; - else if( credits < SHRT_MIN ) - credits = -30000.0f; - } - - G_AddCreditToClient( ent->client, (short)credits, qtrue ); - } - - if( give_all || Q_stricmp( name, "stamina" ) == 0 ) - ent->client->ps.stats[ STAT_STAMINA ] = STAMINA_MAX; - - if( Q_stricmp( name, "poison" ) == 0 ) - { - if( ent->client->pers.teamSelection == TEAM_HUMANS ) - { - ent->client->ps.stats[ STAT_STATE ] |= SS_POISONED; - ent->client->lastPoisonTime = level.time; - ent->client->lastPoisonClient = ent; - } - else - { - ent->client->ps.stats[ STAT_STATE ] |= SS_BOOSTED; - ent->client->boostedTime = level.time; - } - } - - if( Q_stricmp( name, "gas" ) == 0 ) - { - ent->client->ps.eFlags |= EF_POISONCLOUDED; - ent->client->lastPoisonCloudedTime = level.time; - trap_SendServerCommand( ent->client->ps.clientNum, - "poisoncloud" ); - } - - if( give_all || Q_stricmp( name, "ammo" ) == 0 ) - { - gclient_t *client = ent->client; - - if( client->ps.weapon != WP_ALEVEL3_UPG && - BG_Weapon( client->ps.weapon )->infiniteAmmo ) - return; - - client->ps.ammo = BG_Weapon( client->ps.weapon )->maxAmmo; - client->ps.clips = BG_Weapon( client->ps.weapon )->maxClips; - - if( BG_Weapon( client->ps.weapon )->usesEnergy && - BG_InventoryContainsUpgrade( UP_BATTPACK, client->ps.stats ) ) - client->ps.ammo = (int)( (float)client->ps.ammo * BATTPACK_MODIFIER ); - } - - if( give_all || Q_stricmp( name, "fuel" ) == 0 ) - { - if( BG_InventoryContainsUpgrade( UP_JETPACK, ent->client->ps.stats ) ) - ent->client->ps.stats[ STAT_FUEL ] = JETPACK_FUEL_FULL; - } -} - - -/* -================== -Cmd_God_f - -Sets client to godmode - -argv(0) god -================== -*/ -void Cmd_God_f( gentity_t *ent ) -{ - char *msg; - - ent->flags ^= FL_GODMODE; - - if( !( ent->flags & FL_GODMODE ) ) - msg = "godmode OFF\n"; - else - msg = "godmode ON\n"; - - trap_SendServerCommand( ent - g_entities, va( "print \"%s\"", msg ) ); -} - - -/* -================== -Cmd_Notarget_f - -Sets client to notarget - -argv(0) notarget -================== -*/ -void Cmd_Notarget_f( gentity_t *ent ) -{ - char *msg; - - ent->flags ^= FL_NOTARGET; - - if( !( ent->flags & FL_NOTARGET ) ) - msg = "notarget OFF\n"; - else - msg = "notarget ON\n"; - - trap_SendServerCommand( ent - g_entities, va( "print \"%s\"", msg ) ); -} - - -/* -================== -Cmd_Noclip_f - -argv(0) noclip -================== -*/ -void Cmd_Noclip_f( gentity_t *ent ) -{ - char *msg; - - if( ent->client->noclip ) - msg = "noclip OFF\n"; - else - msg = "noclip ON\n"; - - ent->client->noclip = !ent->client->noclip; - - trap_SendServerCommand( ent - g_entities, va( "print \"%s\"", msg ) ); -} - - -/* -================== -Cmd_LevelShot_f - -This is just to help generate the level pictures -for the menus. It goes to the intermission immediately -and sends over a command to the client to resize the view, -hide the scoreboard, and take a special screenshot -================== -*/ -void Cmd_LevelShot_f( gentity_t *ent ) -{ - BeginIntermission( ); - trap_SendServerCommand( ent - g_entities, "clientLevelShot" ); -} - -/* -================= -Cmd_Kill_f -================= -*/ -void Cmd_Kill_f( gentity_t *ent ) -{ - if( g_cheats.integer ) - { - ent->flags &= ~FL_GODMODE; - ent->client->ps.stats[ STAT_HEALTH ] = ent->health = 0; - player_die( ent, ent, ent, 100000, MOD_SUICIDE ); - } - else - { - if( ent->suicideTime == 0 ) - { - trap_SendServerCommand( ent-g_entities, "print \"You will suicide in 20 seconds\n\"" ); - ent->suicideTime = level.time + 20000; - } - else if( ent->suicideTime > level.time ) - { - trap_SendServerCommand( ent-g_entities, "print \"Suicide cancelled\n\"" ); - ent->suicideTime = 0; - } - } -} - -/* -================= -Cmd_Team_f -================= -*/ -void Cmd_Team_f( gentity_t *ent ) -{ - team_t team; - team_t oldteam = ent->client->pers.teamSelection; - char s[ MAX_TOKEN_CHARS ]; - qboolean force = G_admin_permission( ent, ADMF_FORCETEAMCHANGE ); - int aliens = level.numAlienClients; - int humans = level.numHumanClients; - - if( oldteam == TEAM_ALIENS ) - aliens--; - else if( oldteam == TEAM_HUMANS ) - humans--; - - // stop team join spam - if( ent->client->pers.teamChangeTime && - level.time - ent->client->pers.teamChangeTime < 1000 ) - return; - - // stop switching teams for gameplay exploit reasons by enforcing a long - // wait before they can come back - if( !force && !g_cheats.integer && ent->client->pers.aliveSeconds && - level.time - ent->client->pers.teamChangeTime < 30000 ) - { - trap_SendServerCommand( ent-g_entities, - va( "print \"You must wait another %d seconds before changing teams again\n\"", - (int) ( ( 30000 - ( level.time - ent->client->pers.teamChangeTime ) ) / 1000.f ) ) ); - return; - } - - // disallow joining teams during warmup - if( g_doWarmup.integer && ( ( level.warmupTime - level.time ) / 1000 ) > 0 ) - { - G_TriggerMenu( ent - g_entities, MN_WARMUP ); - return; - } - - trap_Argv( 1, s, sizeof( s ) ); - - if( !s[ 0 ] ) - { - trap_SendServerCommand( ent-g_entities, va( "print \"team: %s\n\"", - BG_TeamName( oldteam ) ) ); - return; - } - - if( !Q_stricmp( s, "auto" ) ) - { - if( level.humanTeamLocked && level.alienTeamLocked ) - team = TEAM_NONE; - else if( level.humanTeamLocked || humans > aliens ) - team = TEAM_ALIENS; - - else if( level.alienTeamLocked || aliens > humans ) - team = TEAM_HUMANS; - else - team = TEAM_ALIENS + rand( ) / ( RAND_MAX / 2 + 1 ); - } - else switch( G_TeamFromString( s ) ) - { - case TEAM_NONE: - team = TEAM_NONE; - break; - - case TEAM_ALIENS: - if( level.alienTeamLocked ) - { - G_TriggerMenu( ent - g_entities, MN_A_TEAMLOCKED ); - return; - } - else if( level.humanTeamLocked ) - force = qtrue; - - if( !force && g_teamForceBalance.integer && aliens > humans ) - { - G_TriggerMenu( ent - g_entities, MN_A_TEAMFULL ); - return; - } - - team = TEAM_ALIENS; - break; - - case TEAM_HUMANS: - if( level.humanTeamLocked ) - { - G_TriggerMenu( ent - g_entities, MN_H_TEAMLOCKED ); - return; - } - else if( level.alienTeamLocked ) - force = qtrue; - - if( !force && g_teamForceBalance.integer && humans > aliens ) - { - G_TriggerMenu( ent - g_entities, MN_H_TEAMFULL ); - return; - } - - team = TEAM_HUMANS; - break; - - default: - trap_SendServerCommand( ent-g_entities, - va( "print \"Unknown team: %s\n\"", s ) ); - return; - } - - // stop team join spam - if( oldteam == team ) - return; - - if( team != TEAM_NONE && g_maxGameClients.integer && - level.numPlayingClients >= g_maxGameClients.integer ) - { - G_TriggerMenu( ent - g_entities, MN_PLAYERLIMIT ); - return; - } - - // Apply the change - G_ChangeTeam( ent, team ); -} - -/* -================== -G_CensorString -================== -*/ -static char censors[ 20000 ]; -static int numcensors; - -void G_LoadCensors( void ) -{ - char *text_p, *token; - char text[ 20000 ]; - char *term; - int len; - fileHandle_t f; - - numcensors = 0; - - if( !g_censorship.string[ 0 ] ) - return; - - len = trap_FS_FOpenFile( g_censorship.string, &f, FS_READ ); - if( len < 0 ) - { - Com_Printf( S_COLOR_RED "ERROR: Censors file %s doesn't exist\n", - g_censorship.string ); - return; - } - if( len == 0 || len >= sizeof( text ) - 1 ) - { - trap_FS_FCloseFile( f ); - Com_Printf( S_COLOR_RED "ERROR: Censors file %s is %s\n", - g_censorship.string, len == 0 ? "empty" : "too long" ); - return; - } - trap_FS_Read( text, len, f ); - trap_FS_FCloseFile( f ); - text[ len ] = 0; - - term = censors; - - text_p = text; - while( 1 ) - { - token = COM_Parse( &text_p ); - if( !*token || sizeof( censors ) - ( term - censors ) < 4 ) - break; - Q_strncpyz( term, token, sizeof( censors ) - ( term - censors ) ); - Q_strlwr( term ); - term += strlen( term ) + 1; - if( sizeof( censors ) - ( term - censors ) == 0 ) - break; - token = COM_ParseExt( &text_p, qfalse ); - Q_strncpyz( term, token, sizeof( censors ) - ( term - censors ) ); - term += strlen( term ) + 1; - numcensors++; - } - G_Printf( "Parsed %d string replacements\n", numcensors ); -} - -void G_CensorString( char *out, const char *in, int len, gentity_t *ent ) -{ - const char *s, *m; - int i; - - if( !numcensors || G_admin_permission( ent, ADMF_NOCENSORFLOOD) ) - { - Q_strncpyz( out, in, len ); - return; - } - - len--; - while( *in ) - { - if( Q_IsColorString( in ) ) - { - if( len < 2 ) - break; - *out++ = *in++; - *out++ = *in++; - len -= 2; - continue; - } - if( !isalnum( *in ) ) - { - if( len < 1 ) - break; - *out++ = *in++; - len--; - continue; - } - m = censors; - for( i = 0; i < numcensors; i++, m++ ) - { - s = in; - while( *s && *m ) - { - if( Q_IsColorString( s ) ) - { - s += 2; - continue; - } - if( !isalnum( *s ) ) - { - s++; - continue; - } - if( tolower( *s ) != *m ) - break; - s++; - m++; - } - // match - if( !*m ) - { - in = s; - m++; - while( *m ) - { - if( len < 1 ) - break; - *out++ = *m++; - len--; - } - break; - } - else - { - while( *m ) - m++; - m++; - while( *m ) - m++; - } - } - if( len < 1 ) - break; - // no match - if( i == numcensors ) - { - *out++ = *in++; - len--; - } - } - *out = 0; -} - -/* -================== -G_Say -================== -*/ -static qboolean G_SayTo( gentity_t *ent, gentity_t *other, saymode_t mode, const char *message ) -{ - if( !other ) - return qfalse; - - if( !other->inuse ) - return qfalse; - - if( !other->client ) - return qfalse; - - if( other->client->pers.connected != CON_CONNECTED ) - return qfalse; - - if( ( ent && !OnSameTeam( ent, other ) ) && - ( mode == SAY_TEAM || mode == SAY_AREA || mode == SAY_TPRIVMSG ) ) - { - if( other->client->pers.teamSelection != TEAM_NONE ) - return qfalse; - - // specs with ADMF_SPEC_ALLCHAT flag can see team chat - if( !G_admin_permission( other, ADMF_SPEC_ALLCHAT ) && mode != SAY_TPRIVMSG ) - return qfalse; - } - - trap_SendServerCommand( other-g_entities, va( "chat %d %d \"%s\"", - ent ? ent-g_entities : -1, - mode, - message ) ); - - return qtrue; -} - -void G_Say( gentity_t *ent, saymode_t mode, const char *chatText ) -{ - int j; - gentity_t *other; - // don't let text be too long for malicious reasons - char text[ MAX_SAY_TEXT ]; - - // check if blocked by g_specChat 0 - if( ( !g_specChat.integer ) && ( mode != SAY_TEAM ) && - ( ent ) && ( ent->client->pers.teamSelection == TEAM_NONE ) && - ( !G_admin_permission( ent, ADMF_NOCENSORFLOOD ) ) ) - { - trap_SendServerCommand( ent-g_entities, "print \"say: Global chatting for " - "spectators has been disabled. You may only use team chat.\n\"" ); - mode = SAY_TEAM; - } - - switch( mode ) - { - case SAY_ALL: - G_LogPrintf( "Say: %d \"%s" S_COLOR_WHITE "\": " S_COLOR_GREEN "%s\n", - ( ent ) ? ent - g_entities : -1, - ( ent ) ? ent->client->pers.netname : "console", chatText ); - break; - case SAY_TEAM: - // console say_team is handled in g_svscmds, not here - if( !ent || !ent->client ) - Com_Error( ERR_FATAL, "SAY_TEAM by non-client entity\n" ); - G_LogPrintf( "SayTeam: %d \"%s" S_COLOR_WHITE "\": " S_COLOR_CYAN "%s\n", - ent - g_entities, ent->client->pers.netname, chatText ); - break; - case SAY_RAW: - if( ent ) - Com_Error( ERR_FATAL, "SAY_RAW by client entity\n" ); - G_LogPrintf( "Chat: -1 \"console\": %s\n", chatText ); - default: - break; - } - - G_CensorString( text, chatText, sizeof( text ), ent ); - - // send it to all the apropriate clients - for( j = 0; j < level.maxclients; j++ ) - { - other = &g_entities[ j ]; - G_SayTo( ent, other, mode, text ); - } -} - -/* -================== -Cmd_SayArea_f -================== -*/ -static void Cmd_SayArea_f( gentity_t *ent ) -{ - int entityList[ MAX_GENTITIES ]; - int num, i; - vec3_t range = { 1000.0f, 1000.0f, 1000.0f }; - vec3_t mins, maxs; - char *msg; - - if( trap_Argc( ) < 2 ) - { - ADMP( "usage: say_area [message]\n" ); - return; - } - - msg = ConcatArgs( 1 ); - - for(i = 0; i < 3; i++ ) - range[ i ] = g_sayAreaRange.value; - - G_LogPrintf( "SayArea: %d \"%s" S_COLOR_WHITE "\": " S_COLOR_BLUE "%s\n", - ent - g_entities, ent->client->pers.netname, msg ); - - VectorAdd( ent->s.origin, range, maxs ); - VectorSubtract( ent->s.origin, range, mins ); - - num = trap_EntitiesInBox( mins, maxs, entityList, MAX_GENTITIES ); - for( i = 0; i < num; i++ ) - G_SayTo( ent, &g_entities[ entityList[ i ] ], SAY_AREA, msg ); - - //Send to ADMF_SPEC_ALLCHAT candidates - for( i = 0; i < level.maxclients; i++ ) - { - if( g_entities[ i ].client->pers.teamSelection == TEAM_NONE && - G_admin_permission( &g_entities[ i ], ADMF_SPEC_ALLCHAT ) ) - { - G_SayTo( ent, &g_entities[ i ], SAY_AREA, msg ); - } - } -} - - -/* -================== -Cmd_Say_f -================== -*/ -static void Cmd_Say_f( gentity_t *ent ) -{ - char *p; - char cmd[ MAX_TOKEN_CHARS ]; - saymode_t mode = SAY_ALL; - - if( trap_Argc( ) < 2 ) - return; - - trap_Argv( 0, cmd, sizeof( cmd ) ); - if( Q_stricmp( cmd, "say_team" ) == 0 ) - mode = SAY_TEAM; - - p = ConcatArgs( 1 ); - - G_Say( ent, mode, p ); -} - -/* -================== -Cmd_VSay_f -================== -*/ -void Cmd_VSay_f( gentity_t *ent ) -{ - char arg[MAX_TOKEN_CHARS]; - char text[ MAX_TOKEN_CHARS ]; - voiceChannel_t vchan; - voice_t *voice; - voiceCmd_t *cmd; - voiceTrack_t *track; - int cmdNum = 0; - int trackNum = 0; - char voiceName[ MAX_VOICE_NAME_LEN ] = {"default"}; - char voiceCmd[ MAX_VOICE_CMD_LEN ] = {""}; - char vsay[ 12 ] = {""}; - weapon_t weapon; - - if( !ent || !ent->client ) - Com_Error( ERR_FATAL, "Cmd_VSay_f() called by non-client entity\n" ); - - trap_Argv( 0, arg, sizeof( arg ) ); - if( trap_Argc( ) < 2 ) - { - trap_SendServerCommand( ent-g_entities, va( - "print \"usage: %s command [text] \n\"", arg ) ); - return; - } - if( !level.voices ) - { - trap_SendServerCommand( ent-g_entities, va( - "print \"%s: voice system is not installed on this server\n\"", arg ) ); - return; - } - if( !g_voiceChats.integer ) - { - trap_SendServerCommand( ent-g_entities, va( - "print \"%s: voice system administratively disabled on this server\n\"", - arg ) ); - return; - } - if( !Q_stricmp( arg, "vsay" ) ) - vchan = VOICE_CHAN_ALL; - else if( !Q_stricmp( arg, "vsay_team" ) ) - vchan = VOICE_CHAN_TEAM; - else if( !Q_stricmp( arg, "vsay_local" ) ) - vchan = VOICE_CHAN_LOCAL; - else - return; - Q_strncpyz( vsay, arg, sizeof( vsay ) ); - - if( ent->client->pers.voice[ 0 ] ) - Q_strncpyz( voiceName, ent->client->pers.voice, sizeof( voiceName ) ); - voice = BG_VoiceByName( level.voices, voiceName ); - if( !voice ) - { - trap_SendServerCommand( ent-g_entities, va( - "print \"%s: voice '%s' not found\n\"", vsay, voiceName ) ); - return; - } - - trap_Argv( 1, voiceCmd, sizeof( voiceCmd ) ) ; - cmd = BG_VoiceCmdFind( voice->cmds, voiceCmd, &cmdNum ); - if( !cmd ) - { - trap_SendServerCommand( ent-g_entities, va( - "print \"%s: command '%s' not found in voice '%s'\n\"", - vsay, voiceCmd, voiceName ) ); - return; - } - - // filter non-spec humans by their primary weapon as well - weapon = WP_NONE; - if( ent->client->sess.spectatorState == SPECTATOR_NOT ) - { - weapon = BG_PrimaryWeapon( ent->client->ps.stats ); - } - - track = BG_VoiceTrackFind( cmd->tracks, ent->client->pers.teamSelection, - ent->client->pers.classSelection, weapon, (int)ent->client->voiceEnthusiasm, - &trackNum ); - if( !track ) - { - trap_SendServerCommand( ent-g_entities, va( - "print \"%s: no available track for command '%s', team %d, " - "class %d, weapon %d, and enthusiasm %d in voice '%s'\n\"", - vsay, voiceCmd, ent->client->pers.teamSelection, - ent->client->pers.classSelection, weapon, - (int)ent->client->voiceEnthusiasm, voiceName ) ); - return; - } - - if( !Q_stricmp( ent->client->lastVoiceCmd, cmd->cmd ) ) - ent->client->voiceEnthusiasm++; - - Q_strncpyz( ent->client->lastVoiceCmd, cmd->cmd, - sizeof( ent->client->lastVoiceCmd ) ); - - // optional user supplied text - trap_Argv( 2, arg, sizeof( arg ) ); - G_CensorString( text, arg, sizeof( text ), ent ); - - switch( vchan ) - { - case VOICE_CHAN_ALL: - case VOICE_CHAN_LOCAL: - trap_SendServerCommand( -1, va( - "voice %d %d %d %d \"%s\"\n", - ent-g_entities, vchan, cmdNum, trackNum, text ) ); - break; - case VOICE_CHAN_TEAM: - G_TeamCommand( ent->client->pers.teamSelection, va( - "voice %d %d %d %d \"%s\"\n", - ent-g_entities, vchan, cmdNum, trackNum, text ) ); - break; - default: - break; - } -} - -/* -================== -Cmd_Where_f -================== -*/ -void Cmd_Where_f( gentity_t *ent ) -{ - if( !ent->client ) - return; - trap_SendServerCommand( ent - g_entities, - va( "print \"origin: %f %f %f\n\"", - ent->s.origin[ 0 ], ent->s.origin[ 1 ], - ent->s.origin[ 2 ] ) ); -} - -/* -================== -Cmd_CallVote_f -================== -*/ -void Cmd_CallVote_f( gentity_t *ent ) -{ - char cmd[ MAX_TOKEN_CHARS ], - vote[ MAX_TOKEN_CHARS ], - arg[ MAX_TOKEN_CHARS ]; - char name[ MAX_NAME_LENGTH ] = ""; - char caller[ MAX_NAME_LENGTH ] = ""; - char reason[ MAX_TOKEN_CHARS ]; - char *creason; - int clientNum = -1; - int id = -1; - team_t team; - - trap_Argv( 0, cmd, sizeof( cmd ) ); - trap_Argv( 1, vote, sizeof( vote ) ); - trap_Argv( 2, arg, sizeof( arg ) ); - creason = ConcatArgs( 3 ); - G_DecolorString( creason, reason, sizeof( reason ) ); - - if( !Q_stricmp( cmd, "callteamvote" ) ) - team = ent->client->pers.teamSelection; - else - team = TEAM_NONE; - - if( !g_allowVote.integer ) - { - trap_SendServerCommand( ent-g_entities, - va( "print \"%s: voting not allowed here\n\"", cmd ) ); - return; - } - - if( level.voteTime[ team ] ) - { - trap_SendServerCommand( ent-g_entities, - va( "print \"%s: a vote is already in progress\n\"", cmd ) ); - return; - } - - if( level.voteExecuteTime[ team ] ) - G_ExecuteVote( team ); - - level.voteDelay[ team ] = 0; - level.voteThreshold[ team ] = 50; - - if( g_voteLimit.integer > 0 && - ent->client->pers.namelog->voteCount >= g_voteLimit.integer && - !G_admin_permission( ent, ADMF_NO_VOTE_LIMIT ) ) - { - trap_SendServerCommand( ent-g_entities, va( - "print \"%s: you have already called the maximum number of votes (%d)\n\"", - cmd, g_voteLimit.integer ) ); - return; - } - - // kick, mute, unmute, denybuild, allowbuild - if( !Q_stricmp( vote, "kick" ) || - !Q_stricmp( vote, "mute" ) || !Q_stricmp( vote, "unmute" ) || - !Q_stricmp( vote, "denybuild" ) || !Q_stricmp( vote, "allowbuild" ) ) - { - char err[ MAX_STRING_CHARS ]; - - if( !arg[ 0 ] ) - { - trap_SendServerCommand( ent-g_entities, - va( "print \"%s: no target\n\"", cmd ) ); - return; - } - - // with a little extra work only players from the right team are considered - clientNum = G_ClientNumberFromString( arg, err, sizeof( err ) ); - - if( clientNum == -1 ) - { - ADMP( va( "%s: %s", cmd, err ) ); - return; - } - - G_DecolorString( level.clients[ clientNum ].pers.netname, name, sizeof( name ) ); - id = level.clients[ clientNum ].pers.namelog->id; - - if( !Q_stricmp( vote, "kick" ) || !Q_stricmp( vote, "mute" ) || - !Q_stricmp( vote, "denybuild" ) ) - { - if( G_admin_permission( g_entities + clientNum, ADMF_IMMUNITY ) ) - { - trap_SendServerCommand( ent-g_entities, - va( "print \"%s: admin is immune\n\"", cmd ) ); - - G_AdminMessage( NULL, va( S_COLOR_WHITE "%s" S_COLOR_YELLOW " attempted %s %s" - " on immune admin " S_COLOR_WHITE "%s" S_COLOR_YELLOW - " for: %s", - ent->client->pers.netname, cmd, vote, - g_entities[ clientNum ].client->pers.netname, - reason[ 0 ] ? reason : "no reason" ) ); - return; - } - - if( team != TEAM_NONE && - ( ent->client->pers.teamSelection != - level.clients[ clientNum ].pers.teamSelection ) ) - { - trap_SendServerCommand( ent-g_entities, - va( "print \"%s: player is not on your team\n\"", cmd ) ); - return; - } - - if( !reason[ 0 ] && !G_admin_permission( ent, ADMF_UNACCOUNTABLE ) ) - { - trap_SendServerCommand( ent-g_entities, - va( "print \"%s: You must provide a reason\n\"", cmd ) ); - return; - } - } - } - - if( !Q_stricmp( vote, "kick" ) ) - { - if( level.clients[ clientNum ].pers.localClient ) - { - trap_SendServerCommand( ent-g_entities, - va( "print \"%s: admin is immune\n\"", cmd ) ); - return; - } - - Com_sprintf( level.voteString[ team ], sizeof( level.voteString[ team ] ), - "ban %s \"1s%s\" vote kick (%s)", level.clients[ clientNum ].pers.ip.str, - g_adminTempBan.string, reason ); - Com_sprintf( level.voteDisplayString[ team ], - sizeof( level.voteDisplayString[ team ] ), "Kick player '%s'", name ); - if( reason[ 0 ] ) - { - Q_strcat( level.voteDisplayString[ team ], - sizeof( level.voteDisplayString[ team ] ), va( " for '%s'", reason ) ); - } - } - else if( team == TEAM_NONE ) - { - if( !Q_stricmp( vote, "mute" ) ) - { - if( level.clients[ clientNum ].pers.namelog->muted ) - { - trap_SendServerCommand( ent-g_entities, - va( "print \"%s: player is already muted\n\"", cmd ) ); - return; - } - - Com_sprintf( level.voteString[ team ], sizeof( level.voteString[ team ] ), - "mute %d", id ); - Com_sprintf( level.voteDisplayString[ team ], - sizeof( level.voteDisplayString[ team ] ), - "Mute player '%s'", name ); - if( reason[ 0 ] ) - { - Q_strcat( level.voteDisplayString[ team ], - sizeof( level.voteDisplayString[ team ] ), va( " for '%s'", reason ) ); - } - } - else if( !Q_stricmp( vote, "unmute" ) ) - { - if( !level.clients[ clientNum ].pers.namelog->muted ) - { - trap_SendServerCommand( ent-g_entities, - va( "print \"%s: player is not currently muted\n\"", cmd ) ); - return; - } - - Com_sprintf( level.voteString[ team ], sizeof( level.voteString[ team ] ), - "unmute %d", id ); - Com_sprintf( level.voteDisplayString[ team ], - sizeof( level.voteDisplayString[ team ] ), - "Unmute player '%s'", name ); - } - else if( !Q_stricmp( vote, "map_restart" ) ) - { - strcpy( level.voteString[ team ], vote ); - strcpy( level.voteDisplayString[ team ], "Restart current map" ); - // map_restart comes with a default delay - } - else if( !Q_stricmp( vote, "map" ) ) - { - if( !G_MapExists( arg ) ) - { - trap_SendServerCommand( ent-g_entities, - va( "print \"%s: 'maps/%s.bsp' could not be found on the server\n\"", - cmd, arg ) ); - return; - } - - Com_sprintf( level.voteString[ team ], sizeof( level.voteString ), - "%s \"%s\"", vote, arg ); - Com_sprintf( level.voteDisplayString[ team ], - sizeof( level.voteDisplayString[ team ] ), - "Change to map '%s'", arg ); - level.voteDelay[ team ] = 3000; - } - else if( !Q_stricmp( vote, "nextmap" ) ) - { - if( G_MapExists( g_nextMap.string ) ) - { - trap_SendServerCommand( ent-g_entities, - va( "print \"%s: the next map is already set to '%s'\n\"", - cmd, g_nextMap.string ) ); - return; - } - - if( !G_MapExists( arg ) ) - { - trap_SendServerCommand( ent-g_entities, - va( "print \"%s: 'maps/%s.bsp' could not be found on the server\n\"", - cmd, arg ) ); - return; - } - - Com_sprintf( level.voteString[ team ], sizeof( level.voteString[ team ] ), - "set g_nextMap \"%s\"", arg ); - Com_sprintf( level.voteDisplayString[ team ], - sizeof( level.voteDisplayString[ team ] ), - "Set the next map to '%s'", arg ); - } - else if( !Q_stricmp( vote, "draw" ) ) - { - strcpy( level.voteString[ team ], "evacuation" ); - strcpy( level.voteDisplayString[ team ], "End match in a draw" ); - level.voteDelay[ team ] = 3000; - } - else if( !Q_stricmp( vote, "sudden_death" ) ) - { - if(!g_suddenDeathVotePercent.integer) - { - trap_SendServerCommand( ent-g_entities, - "print \"Sudden Death votes have been disabled\n\"" ); - return; - } - if( G_TimeTilSuddenDeath( ) <= 0 ) - { - trap_SendServerCommand( ent - g_entities, - va( "print \"callvote: Sudden Death has already begun\n\"") ); - return; - } - if( level.suddenDeathBeginTime > 0 && - G_TimeTilSuddenDeath() <= g_suddenDeathVoteDelay.integer * 1000 ) - { - trap_SendServerCommand( ent - g_entities, - va( "print \"callvote: Sudden Death is imminent\n\"") ); - return; - } - level.voteThreshold[ team ] = g_suddenDeathVotePercent.integer; - Com_sprintf( level.voteString[ team ], sizeof( level.voteString[ team ] ), - "suddendeath %d", g_suddenDeathVoteDelay.integer ); - Com_sprintf( level.voteDisplayString[ team ], - sizeof( level.voteDisplayString[ team ] ), - "Begin sudden death in %d seconds", - g_suddenDeathVoteDelay.integer ); - } - else - { - trap_SendServerCommand( ent-g_entities, "print \"Invalid vote string\n\"" ); - trap_SendServerCommand( ent-g_entities, "print \"Valid vote commands are: " - "map, nextmap, map_restart, draw, sudden_death, kick, mute and unmute\n" ); - return; - } - } - else if( !Q_stricmp( vote, "denybuild" ) ) - { - if( level.clients[ clientNum ].pers.namelog->denyBuild ) - { - trap_SendServerCommand( ent-g_entities, - va( "print \"%s: player already lost building rights\n\"", cmd ) ); - return; - } - - Com_sprintf( level.voteString[ team ], sizeof( level.voteString[ team ] ), - "denybuild %d", id ); - Com_sprintf( level.voteDisplayString[ team ], - sizeof( level.voteDisplayString[ team ] ), - "Take away building rights from '%s'", name ); - if( reason[ 0 ] ) - { - Q_strcat( level.voteDisplayString[ team ], - sizeof( level.voteDisplayString[ team ] ), va( " for '%s'", reason ) ); - } - } - else if( !Q_stricmp( vote, "allowbuild" ) ) - { - if( !level.clients[ clientNum ].pers.namelog->denyBuild ) - { - trap_SendServerCommand( ent-g_entities, - va( "print \"%s: player already has building rights\n\"", cmd ) ); - return; - } - - Com_sprintf( level.voteString[ team ], sizeof( level.voteString[ team ] ), - "allowbuild %d", id ); - Com_sprintf( level.voteDisplayString[ team ], - sizeof( level.voteDisplayString[ team ] ), - "Allow '%s' to build", name ); - } - else if( !Q_stricmp( vote, "admitdefeat" ) ) - { - Com_sprintf( level.voteString[ team ], sizeof( level.voteString[ team ] ), - "admitdefeat %d", team ); - strcpy( level.voteDisplayString[ team ], "Admit Defeat" ); - level.voteDelay[ team ] = 3000; - } - else - { - trap_SendServerCommand( ent-g_entities, "print \"Invalid vote string\n\"" ); - trap_SendServerCommand( ent-g_entities, - "print \"Valid team vote commands are: " - "kick, denybuild, allowbuild and admitdefeat\n\"" ); - return; - } - - G_LogPrintf( "%s: %d \"%s" S_COLOR_WHITE "\": %s\n", - team == TEAM_NONE ? "CallVote" : "CallTeamVote", - ent - g_entities, ent->client->pers.netname, level.voteString[ team ] ); - - if( team == TEAM_NONE ) - { - trap_SendServerCommand( -1, va( "print \"%s" S_COLOR_WHITE - " called a vote: %s\n\"", ent->client->pers.netname, - level.voteDisplayString[ team ] ) ); - } - else - { - int i; - - for( i = 0 ; i < level.maxclients ; i++ ) - { - if( level.clients[ i ].pers.connected == CON_CONNECTED ) - { - if( level.clients[ i ].pers.teamSelection == team || - ( level.clients[ i ].pers.teamSelection == TEAM_NONE && - G_admin_permission( &g_entities[ i ], ADMF_SPEC_ALLCHAT ) ) ) - { - trap_SendServerCommand( i, va( "print \"%s" S_COLOR_WHITE - " called a team vote: %s\n\"", ent->client->pers.netname, - level.voteDisplayString[ team ] ) ); - } - else if( G_admin_permission( &g_entities[ i ], ADMF_ADMINCHAT ) ) - { - trap_SendServerCommand( i, va( "chat -1 %d \"" S_COLOR_YELLOW "%s" - S_COLOR_YELLOW " called a team vote (%ss): %s\"", SAY_ADMINS, - ent->client->pers.netname, BG_TeamName( team ), - level.voteDisplayString[ team ] ) ); - } - } - } - } - - G_DecolorString( ent->client->pers.netname, caller, sizeof( caller ) ); - - level.voteTime[ team ] = level.time; - trap_SetConfigstring( CS_VOTE_TIME + team, - va( "%d", level.voteTime[ team ] ) ); - trap_SetConfigstring( CS_VOTE_STRING + team, - level.voteDisplayString[ team ] ); - trap_SetConfigstring( CS_VOTE_CALLER + team, - caller ); - - ent->client->pers.namelog->voteCount++; - ent->client->pers.vote |= 1 << team; - G_Vote( ent, team, qtrue ); -} - -/* -================== -Cmd_Vote_f -================== -*/ -void Cmd_Vote_f( gentity_t *ent ) -{ - char cmd[ MAX_TOKEN_CHARS ], vote[ MAX_TOKEN_CHARS ]; - team_t team = ent->client->pers.teamSelection; - - trap_Argv( 0, cmd, sizeof( cmd ) ); - if( Q_stricmp( cmd, "teamvote" ) ) - team = TEAM_NONE; - - if( !level.voteTime[ team ] ) - { - trap_SendServerCommand( ent-g_entities, - va( "print \"%s: no vote in progress\n\"", cmd ) ); - return; - } - - if( ent->client->pers.voted & ( 1 << team ) ) - { - trap_SendServerCommand( ent-g_entities, - va( "print \"%s: vote already cast\n\"", cmd ) ); - return; - } - - trap_SendServerCommand( ent-g_entities, - va( "print \"%s: vote cast\n\"", cmd ) ); - - trap_Argv( 1, vote, sizeof( vote ) ); - if( vote[ 0 ] == 'y' ) - ent->client->pers.vote |= 1 << team; - else - ent->client->pers.vote &= ~( 1 << team ); - G_Vote( ent, team, qtrue ); -} - - -/* -================= -Cmd_SetViewpos_f -================= -*/ -void Cmd_SetViewpos_f( gentity_t *ent ) -{ - vec3_t origin, angles; - char buffer[ MAX_TOKEN_CHARS ]; - int i; - - if( trap_Argc( ) != 5 ) - { - trap_SendServerCommand( ent-g_entities, "print \"usage: setviewpos x y z yaw\n\"" ); - return; - } - - VectorClear( angles ); - - for( i = 0; i < 3; i++ ) - { - trap_Argv( i + 1, buffer, sizeof( buffer ) ); - origin[ i ] = atof( buffer ); - } - - trap_Argv( 4, buffer, sizeof( buffer ) ); - angles[ YAW ] = atof( buffer ); - - TeleportPlayer( ent, origin, angles ); -} - -#define AS_OVER_RT3 ((ALIENSENSE_RANGE*0.5f)/M_ROOT3) - -static qboolean G_RoomForClassChange( gentity_t *ent, class_t class, - vec3_t newOrigin ) -{ - vec3_t fromMins, fromMaxs; - vec3_t toMins, toMaxs; - vec3_t temp; - trace_t tr; - float nudgeHeight; - float maxHorizGrowth; - class_t oldClass = ent->client->ps.stats[ STAT_CLASS ]; - - BG_ClassBoundingBox( oldClass, fromMins, fromMaxs, NULL, NULL, NULL ); - BG_ClassBoundingBox( class, toMins, toMaxs, NULL, NULL, NULL ); - - VectorCopy( ent->s.origin, newOrigin ); - - // find max x/y diff - maxHorizGrowth = toMaxs[ 0 ] - fromMaxs[ 0 ]; - if( toMaxs[ 1 ] - fromMaxs[ 1 ] > maxHorizGrowth ) - maxHorizGrowth = toMaxs[ 1 ] - fromMaxs[ 1 ]; - if( toMins[ 0 ] - fromMins[ 0 ] > -maxHorizGrowth ) - maxHorizGrowth = -( toMins[ 0 ] - fromMins[ 0 ] ); - if( toMins[ 1 ] - fromMins[ 1 ] > -maxHorizGrowth ) - maxHorizGrowth = -( toMins[ 1 ] - fromMins[ 1 ] ); - - if( maxHorizGrowth > 0.0f ) - { - // test by moving the player up the max required on a 60 degree slope - nudgeHeight = maxHorizGrowth * 2.0f; - } - else - { - // player is shrinking, so there's no need to nudge them upwards - nudgeHeight = 0.0f; - } - - // find what the new origin would be on a level surface - newOrigin[ 2 ] -= toMins[ 2 ] - fromMins[ 2 ]; - - //compute a place up in the air to start the real trace - VectorCopy( newOrigin, temp ); - temp[ 2 ] += nudgeHeight; - trap_Trace( &tr, newOrigin, toMins, toMaxs, temp, ent->s.number, MASK_PLAYERSOLID ); - - //trace down to the ground so that we can evolve on slopes - VectorCopy( newOrigin, temp ); - temp[ 2 ] += ( nudgeHeight * tr.fraction ); - trap_Trace( &tr, temp, toMins, toMaxs, newOrigin, ent->s.number, MASK_PLAYERSOLID ); - VectorCopy( tr.endpos, newOrigin ); - - //make REALLY sure - trap_Trace( &tr, newOrigin, toMins, toMaxs, newOrigin, - ent->s.number, MASK_PLAYERSOLID ); - - //check there is room to evolve - return ( !tr.startsolid && tr.fraction == 1.0f ); -} - -/* -================= -Cmd_Class_f -================= -*/ -void Cmd_Class_f( gentity_t *ent ) -{ - char s[ MAX_TOKEN_CHARS ]; - int clientNum; - int i; - vec3_t infestOrigin; - class_t currentClass = ent->client->pers.classSelection; - class_t newClass; - int entityList[ MAX_GENTITIES ]; - vec3_t range = { AS_OVER_RT3, AS_OVER_RT3, AS_OVER_RT3 }; - vec3_t mins, maxs; - int num; - gentity_t *other; - int oldBoostTime = -1; - vec3_t oldVel; - - clientNum = ent->client - level.clients; - trap_Argv( 1, s, sizeof( s ) ); - newClass = BG_ClassByName( s )->number; - - if( ent->client->sess.spectatorState != SPECTATOR_NOT ) - { - if( ent->client->sess.spectatorState == SPECTATOR_FOLLOW ) - G_StopFollowing( ent ); - if( ent->client->pers.teamSelection == TEAM_ALIENS ) - { - if( newClass != PCL_ALIEN_BUILDER0 && - newClass != PCL_ALIEN_BUILDER0_UPG && - newClass != PCL_ALIEN_LEVEL0 ) - { - G_TriggerMenuArgs( ent->client->ps.clientNum, MN_A_CLASSNOTSPAWN, newClass ); - return; - } - - if( !BG_ClassIsAllowed( newClass ) ) - { - G_TriggerMenuArgs( ent->client->ps.clientNum, MN_A_CLASSNOTALLOWED, newClass ); - return; - } - - if( !BG_ClassAllowedInStage( newClass, g_alienStage.integer ) ) - { - G_TriggerMenuArgs( ent->client->ps.clientNum, MN_A_CLASSNOTATSTAGE, newClass ); - return; - } - - // spawn from an egg - if( G_PushSpawnQueue( &level.alienSpawnQueue, clientNum ) ) - { - ent->client->pers.classSelection = newClass; - ent->client->ps.stats[ STAT_CLASS ] = newClass; - } - } - else if( ent->client->pers.teamSelection == TEAM_HUMANS ) - { - //set the item to spawn with - if( !Q_stricmp( s, BG_Weapon( WP_MACHINEGUN )->name ) && - BG_WeaponIsAllowed( WP_MACHINEGUN ) ) - { - ent->client->pers.humanItemSelection = WP_MACHINEGUN; - } - else if( !Q_stricmp( s, BG_Weapon( WP_HBUILD )->name ) && - BG_WeaponIsAllowed( WP_HBUILD ) ) - { - ent->client->pers.humanItemSelection = WP_HBUILD; - } - else - { - G_TriggerMenu( ent->client->ps.clientNum, MN_H_UNKNOWNSPAWNITEM ); - return; - } - // spawn from a telenode - if( G_PushSpawnQueue( &level.humanSpawnQueue, clientNum ) ) - { - ent->client->pers.classSelection = PCL_HUMAN; - ent->client->ps.stats[ STAT_CLASS ] = PCL_HUMAN; - } - } - return; - } - - if( ent->health <= 0 ) - return; - - if( ent->client->pers.teamSelection == TEAM_ALIENS ) - { - if( newClass == PCL_NONE ) - { - G_TriggerMenu( ent->client->ps.clientNum, MN_A_UNKNOWNCLASS ); - return; - } - - //if we are not currently spectating, we are attempting evolution - if( ent->client->pers.classSelection != PCL_NONE ) - { - int cost; - - //check that we have an overmind - if( !G_Overmind( ) ) - { - G_TriggerMenu( clientNum, MN_A_NOOVMND_EVOLVE ); - return; - } - - //check there are no humans nearby - VectorAdd( ent->client->ps.origin, range, maxs ); - VectorSubtract( ent->client->ps.origin, range, mins ); - - num = trap_EntitiesInBox( mins, maxs, entityList, MAX_GENTITIES ); - for( i = 0; i < num; i++ ) - { - other = &g_entities[ entityList[ i ] ]; - - if( ( other->client && other->client->ps.stats[ STAT_TEAM ] == TEAM_HUMANS ) || - ( other->s.eType == ET_BUILDABLE && other->buildableTeam == TEAM_HUMANS && - other->powered ) ) - { - trace_t tr; - - trap_Trace( &tr, ent->client->ps.origin, NULL, NULL, other->s.origin, ent->client->ps.clientNum, MASK_SOLID ); - - if( tr.fraction > 0.99f ) - { - G_TriggerMenu( clientNum, MN_A_TOOCLOSE ); - return; - } - } - } - - //check that we are not wallwalking - if( ent->client->ps.eFlags & EF_WALLCLIMB ) - { - G_TriggerMenu( clientNum, MN_A_EVOLVEWALLWALK ); - return; - } - - //guard against selling the HBUILD weapons exploit - if( ent->client->sess.spectatorState == SPECTATOR_NOT && - ( currentClass == PCL_ALIEN_BUILDER0 || - currentClass == PCL_ALIEN_BUILDER0_UPG ) && - ent->client->buildTimer ) - { - G_TriggerMenu( ent->client->ps.clientNum, MN_A_EVOLVEBUILDTIMER ); - return; - } - - cost = BG_ClassCanEvolveFromTo( currentClass, newClass, - ent->client->pers.credit, - g_alienStage.integer, 0 ); - - if( G_RoomForClassChange( ent, newClass, infestOrigin ) ) - { - if( cost >= 0 ) - { - - ent->client->pers.evolveHealthFraction = (float)ent->client->ps.stats[ STAT_HEALTH ] / - (float)BG_Class( currentClass )->health; - - if( ent->client->pers.evolveHealthFraction < 0.0f ) - ent->client->pers.evolveHealthFraction = 0.0f; - else if( ent->client->pers.evolveHealthFraction > 1.0f ) - ent->client->pers.evolveHealthFraction = 1.0f; - - //remove credit - G_AddCreditToClient( ent->client, -cost, qtrue ); - ent->client->pers.classSelection = newClass; - ClientUserinfoChanged( clientNum, qfalse ); - VectorCopy( infestOrigin, ent->s.pos.trBase ); - VectorCopy( ent->client->ps.velocity, oldVel ); - - if( ent->client->ps.stats[ STAT_STATE ] & SS_BOOSTED ) - oldBoostTime = ent->client->boostedTime; - - ClientSpawn( ent, ent, ent->s.pos.trBase, ent->s.apos.trBase ); - - VectorCopy( oldVel, ent->client->ps.velocity ); - if( oldBoostTime > 0 ) - { - ent->client->boostedTime = oldBoostTime; - ent->client->ps.stats[ STAT_STATE ] |= SS_BOOSTED; - } - } - else - G_TriggerMenuArgs( clientNum, MN_A_CANTEVOLVE, newClass ); - } - else - G_TriggerMenu( clientNum, MN_A_NOEROOM ); - } - } - else if( ent->client->pers.teamSelection == TEAM_HUMANS ) - G_TriggerMenu( clientNum, MN_H_DEADTOCLASS ); -} - - -/* -================= -Cmd_Destroy_f -================= -*/ -void Cmd_Destroy_f( gentity_t *ent ) -{ - vec3_t viewOrigin, forward, end; - trace_t tr; - gentity_t *traceEnt; - char cmd[ 12 ]; - qboolean deconstruct = qtrue; - qboolean lastSpawn = qfalse; - - if( ent->client->pers.namelog->denyBuild ) - { - G_TriggerMenu( ent->client->ps.clientNum, MN_B_REVOKED ); - return; - } - - trap_Argv( 0, cmd, sizeof( cmd ) ); - if( Q_stricmp( cmd, "destroy" ) == 0 ) - deconstruct = qfalse; - - BG_GetClientViewOrigin( &ent->client->ps, viewOrigin ); - AngleVectors( ent->client->ps.viewangles, forward, NULL, NULL ); - VectorMA( viewOrigin, 100, forward, end ); - - trap_Trace( &tr, viewOrigin, NULL, NULL, end, ent->s.number, MASK_PLAYERSOLID ); - traceEnt = &g_entities[ tr.entityNum ]; - - if( tr.fraction < 1.0f && - ( traceEnt->s.eType == ET_BUILDABLE ) && - ( traceEnt->buildableTeam == ent->client->pers.teamSelection ) && - ( ( ent->client->ps.weapon >= WP_ABUILD ) && - ( ent->client->ps.weapon <= WP_HBUILD ) ) ) - { - // Always let the builder prevent the explosion - if( traceEnt->health <= 0 ) - { - G_QueueBuildPoints( traceEnt ); - G_RewardAttackers( traceEnt ); - G_FreeEntity( traceEnt ); - return; - } - - // Cancel deconstruction (unmark) - if( deconstruct && g_markDeconstruct.integer && traceEnt->deconstruct ) - { - traceEnt->deconstruct = qfalse; - return; - } - - // Prevent destruction of the last spawn - if( ent->client->pers.teamSelection == TEAM_ALIENS && - traceEnt->s.modelindex == BA_A_SPAWN ) - { - if( level.numAlienSpawns <= 1 ) - lastSpawn = qtrue; - } - else if( ent->client->pers.teamSelection == TEAM_HUMANS && - traceEnt->s.modelindex == BA_H_SPAWN ) - { - if( level.numHumanSpawns <= 1 ) - lastSpawn = qtrue; - } - - if( lastSpawn && !g_cheats.integer && - !g_markDeconstruct.integer ) - { - G_TriggerMenu( ent->client->ps.clientNum, MN_B_LASTSPAWN ); - return; - } - - // Don't allow destruction of buildables that cannot be rebuilt - if( G_TimeTilSuddenDeath( ) <= 0 ) - { - G_TriggerMenu( ent->client->ps.clientNum, MN_B_SUDDENDEATH ); - return; - } - - if( !g_markDeconstruct.integer || - ( ent->client->pers.teamSelection == TEAM_HUMANS && - !G_FindPower( traceEnt, qtrue ) ) ) - { - if( ent->client->buildTimer ) - { - G_AddEvent( ent, EV_BUILD_DELAY, ent->client->ps.clientNum ); - return; - } - } - - if( traceEnt->health > 0 ) - { - if( !deconstruct ) - { - G_Damage( traceEnt, ent, ent, forward, tr.endpos, - traceEnt->health, 0, MOD_SUICIDE ); - } - else if( g_markDeconstruct.integer && - ( ent->client->pers.teamSelection != TEAM_HUMANS || - G_FindPower( traceEnt , qtrue ) || lastSpawn ) ) - { - traceEnt->deconstruct = qtrue; // Mark buildable for deconstruction - traceEnt->deconstructTime = level.time; - } - else - { - if( !g_cheats.integer && !g_instantBuild.integer ) // add a bit to the build timer - { - ent->client->buildTimer += - BG_Buildable( traceEnt->s.modelindex, traceEnt->cuboidSize )->buildTime / 4; - G_RecalcBuildTimer(ent->client); - } - G_Damage( traceEnt, ent, ent, forward, tr.endpos, - traceEnt->health, 0, MOD_DECONSTRUCT ); - G_FreeEntity( traceEnt ); - } - } - } -} - -/* -================= -Cmd_ActivateItem_f - -Activate an item -================= -*/ -void Cmd_ActivateItem_f( gentity_t *ent ) -{ - char s[ MAX_TOKEN_CHARS ]; - int upgrade, weapon; - - trap_Argv( 1, s, sizeof( s ) ); - - // "weapon" aliased to whatever weapon you have - if( !Q_stricmp( "weapon", s ) ) - { - if( ent->client->ps.weapon == WP_BLASTER && - BG_PlayerCanChangeWeapon( &ent->client->ps ) ) - G_ForceWeaponChange( ent, WP_NONE ); - return; - } - - upgrade = BG_UpgradeByName( s )->number; - weapon = BG_WeaponByName( s )->number; - - if( upgrade != UP_NONE && BG_InventoryContainsUpgrade( upgrade, ent->client->ps.stats ) ) - BG_ActivateUpgrade( upgrade, ent->client->ps.stats ); - else if( weapon != WP_NONE && - BG_InventoryContainsWeapon( weapon, ent->client->ps.stats ) ) - { - if( ent->client->ps.weapon != weapon && - BG_PlayerCanChangeWeapon( &ent->client->ps ) ) - G_ForceWeaponChange( ent, weapon ); - } - else - trap_SendServerCommand( ent-g_entities, va( "print \"You don't have the %s\n\"", s ) ); -} - - -/* -================= -Cmd_DeActivateItem_f - -Deactivate an item -================= -*/ -void Cmd_DeActivateItem_f( gentity_t *ent ) -{ - char s[ MAX_TOKEN_CHARS ]; - upgrade_t upgrade; - - trap_Argv( 1, s, sizeof( s ) ); - upgrade = BG_UpgradeByName( s )->number; - - if( BG_InventoryContainsUpgrade( upgrade, ent->client->ps.stats ) ) - { - BG_DeactivateUpgrade( upgrade, ent->client->ps.stats ); - if( upgrade == UP_JETPACK ) - BG_AddPredictableEventToPlayerstate( EV_JETPACK_DEACTIVATE, 0, &ent->client->ps ); - } - else - trap_SendServerCommand( ent-g_entities, va( "print \"You don't have the %s\n\"", s ) ); -} - - -/* -================= -Cmd_ToggleItem_f -================= -*/ -void Cmd_ToggleItem_f( gentity_t *ent ) -{ - char s[ MAX_TOKEN_CHARS ]; - weapon_t weapon; - upgrade_t upgrade; - - trap_Argv( 1, s, sizeof( s ) ); - upgrade = BG_UpgradeByName( s )->number; - weapon = BG_WeaponByName( s )->number; - - if( weapon != WP_NONE ) - { - if( !BG_PlayerCanChangeWeapon( &ent->client->ps ) ) - return; - - //special case to allow switching between - //the blaster and the primary weapon - if( ent->client->ps.weapon != WP_BLASTER ) - weapon = WP_BLASTER; - else - weapon = WP_NONE; - - G_ForceWeaponChange( ent, weapon ); - } - else if( BG_InventoryContainsUpgrade( upgrade, ent->client->ps.stats ) ) - { - if( BG_UpgradeIsActive( upgrade, ent->client->ps.stats ) ) - { - BG_DeactivateUpgrade( upgrade, ent->client->ps.stats ); - if( upgrade == UP_JETPACK ) - BG_AddPredictableEventToPlayerstate( EV_JETPACK_DEACTIVATE, 0, &ent->client->ps ); - } - else - BG_ActivateUpgrade( upgrade, ent->client->ps.stats ); - } - else - trap_SendServerCommand( ent-g_entities, va( "print \"You don't have the %s\n\"", s ) ); -} - -/* -================= -Cmd_Buy_f -================= -*/ -void Cmd_Buy_f( gentity_t *ent ) -{ - char s[ MAX_TOKEN_CHARS ]; - weapon_t weapon; - upgrade_t upgrade; - qboolean energyOnly, sellHelmet = qfalse; - - trap_Argv( 1, s, sizeof( s ) ); - - weapon = BG_WeaponByName( s )->number; - upgrade = BG_UpgradeByName( s )->number; - - if( upgrade == UP_NONE && !Q_stricmp(s, "helmet") ) - { - if( g_humanStage.integer < S2 ) - upgrade = UP_HELMET_MK1; - else - upgrade = UP_HELMET_MK2; - sellHelmet = qtrue; - } - - // Only give energy from reactors or repeaters - if( G_BuildableRange( ent->client->ps.origin, 100, BA_H_ARMOURY ) ) - energyOnly = qfalse; - else if( upgrade == UP_AMMO && - BG_Weapon( ent->client->ps.stats[ STAT_WEAPON ] )->usesEnergy && - ( G_BuildableRange( ent->client->ps.origin, 100, BA_H_REACTOR ) || - G_BuildableRange( ent->client->ps.origin, 100, BA_H_REPEATER ) ) ) - energyOnly = qtrue; - else - { - if( upgrade == UP_AMMO && - BG_Weapon( ent->client->ps.weapon )->usesEnergy ) - G_TriggerMenu( ent->client->ps.clientNum, MN_H_NOENERGYAMMOHERE ); - else - G_TriggerMenu( ent->client->ps.clientNum, MN_H_NOARMOURYHERE ); - return; - } - - if( weapon != WP_NONE ) - { - //already got this? - if( BG_InventoryContainsWeapon( weapon, ent->client->ps.stats ) ) - { - G_TriggerMenu( ent->client->ps.clientNum, MN_H_ITEMHELD ); - return; - } - - // Only humans can buy stuff - if( BG_Weapon( weapon )->team != TEAM_HUMANS ) - { - trap_SendServerCommand( ent-g_entities, "print \"You can't buy alien items\n\"" ); - return; - } - - //are we /allowed/ to buy this? - if( !BG_Weapon( weapon )->purchasable ) - { - trap_SendServerCommand( ent-g_entities, "print \"You can't buy this item\n\"" ); - return; - } - - //are we /allowed/ to buy this? - if( !BG_WeaponAllowedInStage( weapon, g_humanStage.integer ) || !BG_WeaponIsAllowed( weapon ) ) - { - trap_SendServerCommand( ent-g_entities, "print \"You can't buy this item\n\"" ); - return; - } - - //can afford this? - if( BG_Weapon( weapon )->price > (short)ent->client->pers.credit ) - { - G_TriggerMenu( ent->client->ps.clientNum, MN_H_NOFUNDS ); - return; - } - - //have space to carry this? - if( BG_Weapon( weapon )->slots & BG_SlotsForInventory( ent->client->ps.stats ) ) - { - G_TriggerMenu( ent->client->ps.clientNum, MN_H_NOSLOTS ); - return; - } - - // In some instances, weapons can't be changed - if( !BG_PlayerCanChangeWeapon( &ent->client->ps ) ) - return; - - ent->client->ps.stats[ STAT_WEAPON ] = weapon; - ent->client->ps.ammo = BG_Weapon( weapon )->maxAmmo; - ent->client->ps.clips = BG_Weapon( weapon )->maxClips; - - if( BG_Weapon( weapon )->usesEnergy && - BG_InventoryContainsUpgrade( UP_BATTPACK, ent->client->ps.stats ) ) - ent->client->ps.ammo *= BATTPACK_MODIFIER; - - G_ForceWeaponChange( ent, weapon ); - - //set build delay/pounce etc to 0 - ent->client->ps.stats[ STAT_MISC ] = 0; - ent->client->buildTimer = 0; - - //subtract from funds - G_AddCreditToClient( ent->client, -(short)BG_Weapon( weapon )->price, qfalse ); - } - else if( upgrade != UP_NONE ) - { - //already got this? - if( !sellHelmet && BG_InventoryContainsUpgrade( upgrade, ent->client->ps.stats ) ) - { - G_TriggerMenu( ent->client->ps.clientNum, MN_H_ITEMHELD ); - return; - } - - //can afford this? - if( BG_Upgrade( upgrade )->price > (short)ent->client->pers.credit ) - { - G_TriggerMenu( ent->client->ps.clientNum, MN_H_NOFUNDS ); - return; - } - - //have space to carry this? - if( !sellHelmet && BG_Upgrade( upgrade )->slots & BG_SlotsForInventory( ent->client->ps.stats ) ) - { - G_TriggerMenu( ent->client->ps.clientNum, MN_H_NOSLOTS ); - return; - } - - // Only humans can buy stuff - if( BG_Upgrade( upgrade )->team != TEAM_HUMANS ) - { - trap_SendServerCommand( ent-g_entities, "print \"You can't buy alien items\n\"" ); - return; - } - - //are we /allowed/ to buy this? - if( !BG_Upgrade( upgrade )->purchasable ) - { - trap_SendServerCommand( ent-g_entities, "print \"You can't buy this item\n\"" ); - return; - } - - //are we /allowed/ to buy this? - if( !BG_UpgradeAllowedInStage( upgrade, g_humanStage.integer ) || !BG_UpgradeIsAllowed( upgrade ) ) - { - trap_SendServerCommand( ent-g_entities, "print \"You can't buy this item\n\"" ); - return; - } - - if( upgrade == UP_AMMO ) - { - G_GiveClientMaxAmmo( ent, energyOnly ); - if( !energyOnly && BG_InventoryContainsUpgrade( UP_JETPACK, ent->client->ps.stats ) && - ent->client->ps.stats[ STAT_FUEL ] < JETPACK_FUEL_FULL ) - { - G_AddEvent( ent, EV_JETPACK_REFUEL, 0) ; - ent->client->ps.stats[ STAT_FUEL ] = JETPACK_FUEL_FULL; - } - } - else - { - if( upgrade == UP_BATTLESUIT ) - { - vec3_t newOrigin; - - if( !G_RoomForClassChange( ent, PCL_HUMAN_BSUIT, newOrigin ) ) - { - G_TriggerMenu( ent->client->ps.clientNum, MN_H_NOROOMBSUITON ); - return; - } - VectorCopy( newOrigin, ent->client->ps.origin ); - ent->client->ps.stats[ STAT_CLASS ] = PCL_HUMAN_BSUIT; - ent->client->pers.classSelection = PCL_HUMAN_BSUIT; - ent->client->ps.eFlags ^= EF_TELEPORT_BIT; - } - else if( upgrade == UP_JETPACK ) - ent->client->ps.stats[ STAT_FUEL ] = JETPACK_FUEL_FULL; - - if( sellHelmet ) - { - BG_RemoveUpgradeFromInventory( UP_HELMET_MK1, ent->client->ps.stats ); - BG_RemoveUpgradeFromInventory( UP_HELMET_MK2, ent->client->ps.stats ); - } - - //add to inventory - BG_AddUpgradeToInventory( upgrade, ent->client->ps.stats ); - } - - if( upgrade == UP_BATTPACK ) - G_GiveClientMaxAmmo( ent, qtrue ); - - //subtract from funds - G_AddCreditToClient( ent->client, -(short)BG_Upgrade( upgrade )->price, qfalse ); - } - else - G_TriggerMenu( ent->client->ps.clientNum, MN_H_UNKNOWNITEM ); - - //update ClientInfo - ClientUserinfoChanged( ent->client->ps.clientNum, qfalse ); -} - - -/* -================= -Cmd_Sell_f -================= -*/ -void Cmd_Sell_f( gentity_t *ent ) -{ - char s[ MAX_TOKEN_CHARS ]; - int i; - weapon_t weapon; - upgrade_t upgrade; - - trap_Argv( 1, s, sizeof( s ) ); - - //no armoury nearby - if( !G_BuildableRange( ent->client->ps.origin, 100, BA_H_ARMOURY ) ) - { - G_TriggerMenu( ent->client->ps.clientNum, MN_H_NOARMOURYHERE ); - return; - } - - if( !Q_stricmpn( s, "weapon", 6 ) ) - weapon = ent->client->ps.stats[ STAT_WEAPON ]; - else - weapon = BG_WeaponByName( s )->number; - - upgrade = BG_UpgradeByName( s )->number; - - if( !upgrade && !Q_stricmp( s, "helmet" ) ) - { - if( BG_InventoryContainsUpgrade( UP_HELMET_MK1, ent->client->ps.stats ) ) - upgrade = UP_HELMET_MK1; - else if( BG_InventoryContainsUpgrade( UP_HELMET_MK2, ent->client->ps.stats ) ) - upgrade = UP_HELMET_MK2; - } - - if( weapon != WP_NONE ) - { - weapon_t selected = BG_GetPlayerWeapon( &ent->client->ps ); - - if( !BG_PlayerCanChangeWeapon( &ent->client->ps ) ) - return; - - //are we /allowed/ to sell this? - if( !BG_Weapon( weapon )->purchasable ) - { - trap_SendServerCommand( ent-g_entities, "print \"You can't sell this weapon\n\"" ); - return; - } - - //remove weapon if carried - if( BG_InventoryContainsWeapon( weapon, ent->client->ps.stats ) ) - { - //guard against selling the HBUILD weapons exploit - if( weapon == WP_HBUILD && ent->client->buildTimer ) - { - G_TriggerMenu( ent->client->ps.clientNum, MN_H_ARMOURYBUILDTIMER ); - return; - } - - ent->client->ps.stats[ STAT_WEAPON ] = WP_NONE; - // Cancel ghost buildables - ent->client->ps.stats[ STAT_BUILDABLE ] = BA_NONE; - - //add to funds - G_AddCreditToClient( ent->client, (short)BG_Weapon( weapon )->price, qfalse ); - } - - //if we have this weapon selected, force a new selection - if( weapon == selected ) - G_ForceWeaponChange( ent, WP_NONE ); - } - else if( upgrade != UP_NONE ) - { - //are we /allowed/ to sell this? - if( !BG_Upgrade( upgrade )->purchasable ) - { - trap_SendServerCommand( ent-g_entities, "print \"You can't sell this item\n\"" ); - return; - } - //remove upgrade if carried - if( BG_InventoryContainsUpgrade( upgrade, ent->client->ps.stats ) ) - { - // shouldn't really need to test for this, but just to be safe - if( upgrade == UP_BATTLESUIT ) - { - vec3_t newOrigin; - - if( !G_RoomForClassChange( ent, PCL_HUMAN, newOrigin ) ) - { - G_TriggerMenu( ent->client->ps.clientNum, MN_H_NOROOMBSUITOFF ); - return; - } - VectorCopy( newOrigin, ent->client->ps.origin ); - ent->client->ps.stats[ STAT_CLASS ] = PCL_HUMAN; - ent->client->pers.classSelection = PCL_HUMAN; - ent->client->ps.eFlags ^= EF_TELEPORT_BIT; - } - - //add to inventory - BG_RemoveUpgradeFromInventory( upgrade, ent->client->ps.stats ); - - if( upgrade == UP_BATTPACK ) - G_GiveClientMaxAmmo( ent, qtrue ); - - //add to funds - G_AddCreditToClient( ent->client, (short)BG_Upgrade( upgrade )->price, qfalse ); - } - } - else if( !Q_stricmp( s, "weapons" ) ) - { - weapon_t selected = BG_GetPlayerWeapon( &ent->client->ps ); - - if( !BG_PlayerCanChangeWeapon( &ent->client->ps ) ) - return; - - for( i = WP_NONE + 1; i < WP_NUM_WEAPONS; i++ ) - { - //guard against selling the HBUILD weapons exploit - if( i == WP_HBUILD && ent->client->buildTimer ) - { - G_TriggerMenu( ent->client->ps.clientNum, MN_H_ARMOURYBUILDTIMER ); - continue; - } - - if( BG_InventoryContainsWeapon( i, ent->client->ps.stats ) && - BG_Weapon( i )->purchasable ) - { - ent->client->ps.stats[ STAT_WEAPON ] = WP_NONE; - - //add to funds - G_AddCreditToClient( ent->client, (short)BG_Weapon( i )->price, qfalse ); - } - - //if we have this weapon selected, force a new selection - if( i == selected ) - G_ForceWeaponChange( ent, WP_NONE ); - } - } - else if( !Q_stricmp( s, "upgrades" ) ) - { - for( i = UP_NONE + 1; i < UP_NUM_UPGRADES; i++ ) - { - //remove upgrade if carried - if( BG_InventoryContainsUpgrade( i, ent->client->ps.stats ) && - BG_Upgrade( i )->purchasable ) - { - - // shouldn't really need to test for this, but just to be safe - if( i == UP_BATTLESUIT ) - { - vec3_t newOrigin; - - if( !G_RoomForClassChange( ent, PCL_HUMAN, newOrigin ) ) - { - G_TriggerMenu( ent->client->ps.clientNum, MN_H_NOROOMBSUITOFF ); - continue; - } - VectorCopy( newOrigin, ent->client->ps.origin ); - ent->client->ps.stats[ STAT_CLASS ] = PCL_HUMAN; - ent->client->pers.classSelection = PCL_HUMAN; - ent->client->ps.eFlags ^= EF_TELEPORT_BIT; - } - - BG_RemoveUpgradeFromInventory( i, ent->client->ps.stats ); - - if( i == UP_BATTPACK ) - G_GiveClientMaxAmmo( ent, qtrue ); - - //add to funds - G_AddCreditToClient( ent->client, (short)BG_Upgrade( i )->price, qfalse ); - } - } - } - else - G_TriggerMenu( ent->client->ps.clientNum, MN_H_UNKNOWNITEM ); - - //update ClientInfo - ClientUserinfoChanged( ent->client->ps.clientNum, qfalse ); -} - -/* -================= -Cmd_CheckCuboidSize - -Check if the specified dimensions are valid. -================= -*/ -qboolean Cmd_CheckCuboidSize(vec3_t dims) -{ - if(g_cuboidSizeLimit.integer) - if(dims[0]>g_cuboidSizeLimit.integer||dims[1]>g_cuboidSizeLimit.integer||dims[2]>g_cuboidSizeLimit.integer) - return qfalse; - if(dims[0]*dims[1]*dims[2]<CUBOID_MINVOLUME) - return qfalse; - if(dims[0]<1||dims[1]<1||dims[2]<1) - return qfalse; - return qtrue; -} - -/* -================= -Cmd_Cb_f - -Sent by cgame in background (not directly by a person). -Update player's cuboid selection (after validation) with data sent over network. -================= -*/ -void Cmd_Cb_f(gentity_t *ent) -{ - char s[MAX_TOKEN_CHARS]; - int echo; - vec3_t dims; - - if(trap_Argc()!=5) - return; - trap_Argv(1,s,sizeof(s)); - echo=atoi(s); - trap_Argv(2,s,sizeof(s)); - dims[0]=atof(s); - trap_Argv(3,s,sizeof(s)); - dims[1]=atof(s); - trap_Argv(4,s,sizeof(s)); - dims[2]=atof(s); - if(Cmd_CheckCuboidSize(dims)) - { - VectorCopy(dims,ent->client->cuboidSelection); - trap_SendServerCommand(ent->client-level.clients,va("cb3 %i\n",echo)); - G_RelayCuboidToSpectators(ent->client); - } - else - { - if(Cmd_CheckCuboidSize(ent->client->cuboidSelection)) - trap_SendServerCommand(ent->client-level.clients,va("cb3 %i %f %f %f\n",echo,ent->client->cuboidSelection[0],ent->client->cuboidSelection[1],ent->client->cuboidSelection[2])); - else - trap_SendServerCommand(ent->client-level.clients,va("cb3 %i %f %f %f\n",echo,g_cuboidSizeLimit.integer,g_cuboidSizeLimit.integer,g_cuboidSizeLimit.integer)); - } -} - - -/* -================= -Cmd_Build_f -================= -*/ -void Cmd_Build_f( gentity_t *ent ) -{ - char s[ MAX_TOKEN_CHARS ]; - buildable_t buildable; - float dist; - vec3_t origin, normal; - team_t team; - char buf[128]; - vec3_t dims; - - if( ent->client->pers.namelog->denyBuild ) - { - G_TriggerMenu( ent->client->ps.clientNum, MN_B_REVOKED ); - return; - } - - if( ent->client->pers.teamSelection == level.surrenderTeam ) - { - G_TriggerMenu( ent->client->ps.clientNum, MN_B_SURRENDER ); - return; - } - - trap_Argv( 1, s, sizeof( s ) ); - - buildable = BG_BuildableByName( s )->number; - - /* To allow players build cuboids with completely arbitrary - * dimensions (the current resizing method doesn't provide - * much precision) the build command was extended with the - * following syntax: - * build [building] [X] [Y] [Z] - * where X, Y and Z are respectively the cuboid's length - * on the X, Y and Z axis. - */ - if( BG_Buildable(buildable,NULL)->cuboid ) - { - if( trap_Argc() >= 5 ) - { - trap_Argv(2,s,sizeof(s)); - dims[0]=MAX(1,atof(s)); - trap_Argv(3,s,sizeof(s)); - dims[1]=MAX(1,atof(s)); - trap_Argv(4,s,sizeof(s)); - dims[2]=MAX(1,atof(s)); - if(!Cmd_CheckCuboidSize(dims)) - { - Com_sprintf(buf,sizeof(buf),"print \"^1error: invalid cuboid size (min volume: %i, max size: %s)\n\"", - CUBOID_MINVOLUME,(g_cuboidSizeLimit.integer?va("%ix%ix%i",g_cuboidSizeLimit.integer,g_cuboidSizeLimit.integer, g_cuboidSizeLimit.integer):"no limit")); - trap_SendServerCommand(ent->client-level.clients,buf); - return; - } - VectorCopy(dims,ent->client->cuboidSelection); - } - // client is building a cuboid for the first time so reset the selection to default - if(!Cmd_CheckCuboidSize(ent->client->cuboidSelection)) - { - ent->client->cuboidSelection[0]=32; - ent->client->cuboidSelection[1]=32; - ent->client->cuboidSelection[2]=32; - trap_SendServerCommand(ent->client-level.clients,"cb2 32 32 32"); - G_RelayCuboidToSpectators(ent->client); - } - - if(!BG_CuboidAllowed((team==TEAM_ALIENS?g_alienStage.integer:g_humanStage.integer))) - { - if(BG_CuboidMode()==1) - G_TriggerMenu(ent->client->ps.clientNum,MN_B_CUBOID_MODE1); - else - G_TriggerMenu(ent->client->ps.clientNum,MN_B_CUBOID_MODE2); - return; - } - } - - if( G_TimeTilSuddenDeath( ) <= 0 ) - { - G_TriggerMenu( ent->client->ps.clientNum, MN_B_SUDDENDEATH ); - return; - } - - team = ent->client->ps.stats[ STAT_TEAM ]; - - if( buildable != BA_NONE && - ( ( 1 << ent->client->ps.weapon ) & BG_Buildable( buildable, NULL )->buildWeapon ) && - BG_BuildableIsAllowed( buildable ) && - ( ( team == TEAM_ALIENS && BG_BuildableAllowedInStage( buildable, g_alienStage.integer ) ) || - ( team == TEAM_HUMANS && BG_BuildableAllowedInStage( buildable, g_humanStage.integer ) ) ) ) - { - dynMenu_t err; - dist = BG_Class( ent->client->ps.stats[ STAT_CLASS ] )->buildDist; - - ent->client->ps.stats[ STAT_BUILDABLE ] = BA_NONE; - - //these are the errors displayed when the builder first selects something to use - switch( G_CanBuild( ent, buildable, dist, origin, normal, ent->client->cuboidSelection ) ) - { - // can place right away, set the blueprint and the valid togglebit - case IBE_NONE: - case IBE_TNODEWARN: - case IBE_RPTNOREAC: - case IBE_RPTPOWERHERE: - case IBE_SPWNWARN: - err = MN_NONE; - // we OR-in the selected builable later - ent->client->ps.stats[ STAT_BUILDABLE ] = SB_VALID_TOGGLEBIT; - break; - - // can't place yet but maybe soon: start with valid togglebit off - case IBE_NORMAL: - case IBE_NOCREEP: - case IBE_NOROOM: - case IBE_NOOVERMIND: - case IBE_NOPOWERHERE: - case IBE_NOSURF: - err = MN_NONE; - break; - - // more serious errors just pop a menu - case IBE_NOALIENBP: - err = MN_A_NOBP; - break; - - case IBE_ONEOVERMIND: - err = MN_A_ONEOVERMIND; - break; - - case IBE_ONEREACTOR: - err = MN_H_ONEREACTOR; - break; - - case IBE_NOHUMANBP: - err = MN_H_NOBP; - break; - - case IBE_NODCC: - err = MN_H_NODCC; - break; - - case IBE_PERMISSION: - err = MN_B_CANNOT; - break; - - case IBE_LASTSPAWN: - err = MN_B_LASTSPAWN; - break; - - case IBE_TOODENSE: - err = MN_B_TOODENSE; - break; - - default: - err = -1; // stop uninitialised warning - break; - } - - if( ( err == MN_A_NOBP || err == MN_H_NOBP ) && BG_Buildable(buildable,NULL)->cuboid ) - { - err = MN_NONE; - ent->client->ps.stats[ STAT_BUILDABLE ] = SB_VALID_TOGGLEBIT; - } - - if( err == MN_NONE || ent->client->pers.disableBlueprintErrors ) - { - trap_SendServerCommand(ent->client-level.clients,va("cb2 %f %f %f\n", - ent->client->cuboidSelection[0], - ent->client->cuboidSelection[1], - ent->client->cuboidSelection[2])); - G_RelayCuboidToSpectators(ent->client); - ent->client->ps.stats[ STAT_BUILDABLE ] |= buildable; - } - else - G_TriggerMenu( ent->client->ps.clientNum, err ); - } - else - G_TriggerMenu( ent->client->ps.clientNum, MN_B_CANNOT ); -} - -/* -================= -Cmd_Reload_f -================= -*/ -void Cmd_Reload_f( gentity_t *ent ) -{ - playerState_t *ps = &ent->client->ps; - int ammo; - - // weapon doesn't ever need reloading - if( BG_Weapon( ps->weapon )->infiniteAmmo ) - return; - - if( ps->clips <= 0 ) - return; - - if( BG_Weapon( ps->weapon )->usesEnergy && - BG_InventoryContainsUpgrade( UP_BATTPACK, ps->stats ) ) - ammo = BG_Weapon( ps->weapon )->maxAmmo * BATTPACK_MODIFIER; - else - ammo = BG_Weapon( ps->weapon )->maxAmmo; - - // don't reload when full - if( ps->ammo >= ammo ) - return; - - // the animation, ammo refilling etc. is handled by PM_Weapon - if( ent->client->ps.weaponstate != WEAPON_RELOADING ) - ent->client->ps.pm_flags |= PMF_WEAPON_RELOAD; -} - -/* -================= -G_StopFromFollowing - -stops any other clients from following this one -called when a player leaves a team or dies -================= -*/ -void G_StopFromFollowing( gentity_t *ent ) -{ - int i; - - for( i = 0; i < level.maxclients; i++ ) - { - if( level.clients[ i ].sess.spectatorState == SPECTATOR_FOLLOW && - level.clients[ i ].sess.spectatorClient == ent->client->ps.clientNum ) - { - if( !G_FollowNewClient( &g_entities[ i ], 1 ) ) - G_StopFollowing( &g_entities[ i ] ); - } - } -} - -/* -================= -G_StopFollowing - -If the client being followed leaves the game, or you just want to drop -to free floating spectator mode -================= -*/ -void G_StopFollowing( gentity_t *ent ) -{ - ent->client->ps.stats[ STAT_TEAM ] = ent->client->pers.teamSelection; - - if( ent->client->pers.teamSelection == TEAM_NONE ) - { - ent->client->sess.spectatorState = - ent->client->ps.persistant[ PERS_SPECSTATE ] = SPECTATOR_FREE; - } - else - { - vec3_t spawn_origin, spawn_angles; - - ent->client->sess.spectatorState = - ent->client->ps.persistant[ PERS_SPECSTATE ] = SPECTATOR_LOCKED; - - if( ent->client->pers.teamSelection == TEAM_ALIENS ) - G_SelectAlienLockSpawnPoint( spawn_origin, spawn_angles ); - else if( ent->client->pers.teamSelection == TEAM_HUMANS ) - G_SelectHumanLockSpawnPoint( spawn_origin, spawn_angles ); - - G_SetOrigin( ent, spawn_origin ); - VectorCopy( spawn_origin, ent->client->ps.origin ); - G_SetClientViewAngle( ent, spawn_angles ); - } - ent->client->sess.spectatorClient = -1; - ent->client->ps.pm_flags &= ~PMF_FOLLOW; - ent->client->ps.groundEntityNum = ENTITYNUM_NONE; - ent->client->ps.stats[ STAT_STATE ] = 0; - ent->client->ps.stats[ STAT_VIEWLOCK ] = 0; - ent->client->ps.eFlags &= ~( EF_WALLCLIMB | EF_WALLCLIMBCEILING ); - ent->client->ps.viewangles[ PITCH ] = 0.0f; - ent->client->ps.clientNum = ent - g_entities; - ent->client->ps.persistant[ PERS_CREDIT ] = ent->client->pers.credit; - - CalculateRanks( ); -} - -/* -================= -G_FollowLockView - -Client is still following a player, but that player has gone to spectator -mode and cannot be followed for the moment -================= -*/ -void G_FollowLockView( gentity_t *ent ) -{ - vec3_t spawn_origin, spawn_angles; - int clientNum; - - clientNum = ent->client->sess.spectatorClient; - ent->client->sess.spectatorState = - ent->client->ps.persistant[ PERS_SPECSTATE ] = SPECTATOR_FOLLOW; - ent->client->ps.clientNum = clientNum; - ent->client->ps.pm_flags &= ~PMF_FOLLOW; - ent->client->ps.stats[ STAT_TEAM ] = ent->client->pers.teamSelection; - ent->client->ps.stats[ STAT_STATE ] &= ~SS_WALLCLIMBING; - ent->client->ps.stats[ STAT_VIEWLOCK ] = 0; - ent->client->ps.eFlags &= ~( EF_WALLCLIMB | EF_WALLCLIMBCEILING ); - ent->client->ps.eFlags ^= EF_TELEPORT_BIT; - ent->client->ps.viewangles[ PITCH ] = 0.0f; - - // Put the view at the team spectator lock position - if( level.clients[ clientNum ].pers.teamSelection == TEAM_ALIENS ) - G_SelectAlienLockSpawnPoint( spawn_origin, spawn_angles ); - else if( level.clients[ clientNum ].pers.teamSelection == TEAM_HUMANS ) - G_SelectHumanLockSpawnPoint( spawn_origin, spawn_angles ); - - G_SetOrigin( ent, spawn_origin ); - VectorCopy( spawn_origin, ent->client->ps.origin ); - G_SetClientViewAngle( ent, spawn_angles ); -} - -/* -================= -G_FollowNewClient - -This was a really nice, elegant function. Then I fucked it up. -================= -*/ -qboolean G_FollowNewClient( gentity_t *ent, int dir ) -{ - int clientnum = ent->client->sess.spectatorClient; - int original = clientnum; - qboolean selectAny = qfalse; - - if( dir > 1 ) - dir = 1; - else if( dir < -1 ) - dir = -1; - else if( dir == 0 ) - return qtrue; - - if( ent->client->sess.spectatorState == SPECTATOR_NOT ) - return qfalse; - - // select any if no target exists - if( clientnum < 0 || clientnum >= level.maxclients ) - { - clientnum = original = 0; - selectAny = qtrue; - } - - do - { - clientnum += dir; - - if( clientnum >= level.maxclients ) - clientnum = 0; - - if( clientnum < 0 ) - clientnum = level.maxclients - 1; - - // can't follow self - if( &g_entities[ clientnum ] == ent ) - continue; - - // avoid selecting existing follow target - if( clientnum == original && !selectAny ) - continue; //effectively break; - - // can only follow connected clients - if( level.clients[ clientnum ].pers.connected != CON_CONNECTED ) - continue; - - // can't follow a spectator - if( level.clients[ clientnum ].pers.teamSelection == TEAM_NONE ) - continue; - - // if stickyspec is disabled, can't follow someone in queue either - if( !ent->client->pers.stickySpec && - level.clients[ clientnum ].sess.spectatorState != SPECTATOR_NOT ) - continue; - - // can only follow teammates when dead and on a team - if( ent->client->pers.teamSelection != TEAM_NONE && - ( level.clients[ clientnum ].pers.teamSelection != - ent->client->pers.teamSelection ) ) - continue; - - // this is good, we can use it - ent->client->sess.spectatorClient = clientnum; - ent->client->sess.spectatorState = SPECTATOR_FOLLOW; - - // if this client is in the spawn queue, we need to do something special - if( level.clients[ clientnum ].sess.spectatorState != SPECTATOR_NOT ) - G_FollowLockView( ent ); - - return qtrue; - - } while( clientnum != original ); - - return qfalse; -} - -/* -================= -G_ToggleFollow -================= -*/ -void G_ToggleFollow( gentity_t *ent ) -{ - if( ent->client->sess.spectatorState == SPECTATOR_FOLLOW ) - G_StopFollowing( ent ); - else - G_FollowNewClient( ent, 1 ); -} - -/* -================= -Cmd_Follow_f -================= -*/ -void Cmd_Follow_f( gentity_t *ent ) -{ - int i; - char arg[ MAX_NAME_LENGTH ]; - - // won't work unless spectating - if( ent->client->sess.spectatorState == SPECTATOR_NOT ) - return; - - if( trap_Argc( ) != 2 ) - { - G_ToggleFollow( ent ); - } - else - { - char err[ MAX_STRING_CHARS ]; - trap_Argv( 1, arg, sizeof( arg ) ); - - i = G_ClientNumberFromString( arg, err, sizeof( err ) ); - - if( i == -1 ) - { - trap_SendServerCommand( ent - g_entities, - va( "print \"follow: %s\"", err ) ); - return; - } - - // can't follow self - if( &level.clients[ i ] == ent->client ) - return; - - // can't follow another spectator if sticky spec is off - if( !ent->client->pers.stickySpec && - level.clients[ i ].sess.spectatorState != SPECTATOR_NOT ) - return; - - // if not on team spectator, you can only follow teammates - if( ent->client->pers.teamSelection != TEAM_NONE && - ( level.clients[ i ].pers.teamSelection != - ent->client->pers.teamSelection ) ) - return; - - ent->client->sess.spectatorState = SPECTATOR_FOLLOW; - ent->client->sess.spectatorClient = i; - } -} - -/* -================= -Cmd_FollowCycle_f -================= -*/ -void Cmd_FollowCycle_f( gentity_t *ent ) -{ - char args[ 11 ]; - int dir = 1; - - trap_Argv( 0, args, sizeof( args ) ); - if( Q_stricmp( args, "followprev" ) == 0 ) - dir = -1; - - // won't work unless spectating - if( ent->client->sess.spectatorState == SPECTATOR_NOT ) - return; - - G_FollowNewClient( ent, dir ); -} - -static void Cmd_Ignore_f( gentity_t *ent ) -{ - int pids[ MAX_CLIENTS ]; - char name[ MAX_NAME_LENGTH ]; - char cmd[ 9 ]; - int matches = 0; - int i; - qboolean ignore = qfalse; - - trap_Argv( 0, cmd, sizeof( cmd ) ); - if( Q_stricmp( cmd, "ignore" ) == 0 ) - ignore = qtrue; - - if( trap_Argc() < 2 ) - { - trap_SendServerCommand( ent-g_entities, va( "print \"[skipnotify]" - "usage: %s [clientNum | partial name match]\n\"", cmd ) ); - return; - } - - Q_strncpyz( name, ConcatArgs( 1 ), sizeof( name ) ); - matches = G_ClientNumbersFromString( name, pids, MAX_CLIENTS ); - if( matches < 1 ) - { - trap_SendServerCommand( ent-g_entities, va( "print \"[skipnotify]" - "%s: no clients match the name '%s'\n\"", cmd, name ) ); - return; - } - - for( i = 0; i < matches; i++ ) - { - if( ignore ) - { - if( !Com_ClientListContains( &ent->client->sess.ignoreList, pids[ i ] ) ) - { - Com_ClientListAdd( &ent->client->sess.ignoreList, pids[ i ] ); - ClientUserinfoChanged( ent->client->ps.clientNum, qfalse ); - trap_SendServerCommand( ent-g_entities, va( "print \"[skipnotify]" - "ignore: added %s^7 to your ignore list\n\"", - level.clients[ pids[ i ] ].pers.netname ) ); - } - else - { - trap_SendServerCommand( ent-g_entities, va( "print \"[skipnotify]" - "ignore: %s^7 is already on your ignore list\n\"", - level.clients[ pids[ i ] ].pers.netname ) ); - } - } - else - { - if( Com_ClientListContains( &ent->client->sess.ignoreList, pids[ i ] ) ) - { - Com_ClientListRemove( &ent->client->sess.ignoreList, pids[ i ] ); - ClientUserinfoChanged( ent->client->ps.clientNum, qfalse ); - trap_SendServerCommand( ent-g_entities, va( "print \"[skipnotify]" - "unignore: removed %s^7 from your ignore list\n\"", - level.clients[ pids[ i ] ].pers.netname ) ); - } - else - { - trap_SendServerCommand( ent-g_entities, va( "print \"[skipnotify]" - "unignore: %s^7 is not on your ignore list\n\"", - level.clients[ pids[ i ] ].pers.netname ) ); - } - } - } -} - -/* -================= -Cmd_ListMaps_f - -List all maps on the server -================= -*/ - -static int SortMaps( const void *a, const void *b ) -{ - return strcmp( *(char **)a, *(char **)b ); -} - -#define MAX_MAPLIST_MAPS 256 -#define MAX_MAPLIST_ROWS 9 -void Cmd_ListMaps_f( gentity_t *ent ) -{ - char search[ 16 ] = {""}; - char fileList[ 4096 ] = {""}; - char *fileSort[ MAX_MAPLIST_MAPS ]; - char *filePtr, *p; - int numFiles; - int fileLen = 0; - int shown = 0; - int count = 0; - int page = 0; - int pages; - int row, rows; - int start, i, j; - - if( trap_Argc( ) > 1 ) - { - trap_Argv( 1, search, sizeof( search ) ); - for( p = search; ( *p ) && isdigit( *p ); p++ ); - if( !( *p ) ) - { - page = atoi( search ); - search[ 0 ] = '\0'; - } - else if( trap_Argc( ) > 2 ) - { - char lp[ 8 ]; - trap_Argv( 2, lp, sizeof( lp ) ); - page = atoi( lp ); - } - - if( page > 0 ) - page--; - else if( page < 0 ) - page = 0; - } - - numFiles = trap_FS_GetFileList( "maps/", ".bsp", - fileList, sizeof( fileList ) ); - filePtr = fileList; - for( i = 0; i < numFiles && count < MAX_MAPLIST_MAPS; i++, filePtr += fileLen + 1 ) - { - fileLen = strlen( filePtr ); - if ( fileLen < 5 ) - continue; - - filePtr[ fileLen - 4 ] = '\0'; - - if( search[ 0 ] && !strstr( filePtr, search ) ) - continue; - - fileSort[ count ] = filePtr; - count++; - } - qsort( fileSort, count, sizeof( fileSort[ 0 ] ), SortMaps ); - - rows = ( count + 2 ) / 3; - pages = MAX( 1, ( rows + MAX_MAPLIST_ROWS - 1 ) / MAX_MAPLIST_ROWS ); - if( page >= pages ) - page = pages - 1; - - start = page * MAX_MAPLIST_ROWS * 3; - if( count < start + ( 3 * MAX_MAPLIST_ROWS ) ) - rows = ( count - start + 2 ) / 3; - else - rows = MAX_MAPLIST_ROWS; - - ADMBP_begin( ); - for( row = 0; row < rows; row++ ) - { - for( i = start + row, j = 0; i < count && j < 3; i += rows, j++ ) - { - ADMBP( va( "^7 %-20s", fileSort[ i ] ) ); - shown++; - } - ADMBP( "\n" ); - } - if ( search[ 0 ] ) - ADMBP( va( "^3listmaps: ^7found %d maps matching '%s^7'", count, search ) ); - else - ADMBP( va( "^3listmaps: ^7listing %d of %d maps", shown, count ) ); - if( pages > 1 ) - ADMBP( va( ", page %d of %d", page + 1, pages ) ); - if( page + 1 < pages ) - ADMBP( va( ", use 'listmaps %s%s%d' to see more", - search, ( search[ 0 ] ) ? " ": "", page + 2 ) ); - ADMBP( ".\n" ); - ADMBP_end( ); -} - -/* -================= -Cmd_Test_f -================= -*/ -void Cmd_Test_f( gentity_t *humanPlayer ) -{ -} - -/* -================= -Cmd_Damage_f - -Deals damage to you (for testing), arguments: [damage] [dx] [dy] [dz] -The dx/dy arguments describe the damage point's offset from the entity origin -================= -*/ -void Cmd_Damage_f( gentity_t *ent ) -{ - vec3_t point; - char arg[ 16 ]; - float dx = 0.0f, dy = 0.0f, dz = 100.0f; - int damage = 100; - qboolean nonloc = qtrue; - - if( trap_Argc() > 1 ) - { - trap_Argv( 1, arg, sizeof( arg ) ); - damage = atoi( arg ); - } - if( trap_Argc() > 4 ) - { - trap_Argv( 2, arg, sizeof( arg ) ); - dx = atof( arg ); - trap_Argv( 3, arg, sizeof( arg ) ); - dy = atof( arg ); - trap_Argv( 4, arg, sizeof( arg ) ); - dz = atof( arg ); - nonloc = qfalse; - } - VectorCopy( ent->s.origin, point ); - point[ 0 ] += dx; - point[ 1 ] += dy; - point[ 2 ] += dz; - G_Damage( ent, NULL, NULL, NULL, point, damage, - ( nonloc ? DAMAGE_NO_LOCDAMAGE : 0 ), MOD_TARGET_LASER ); -} - -/* -================== -G_FloodLimited - -Determine whether a user is flood limited, and adjust their flood demerits -Print them a warning message if they are over the limit -Return is time in msec until the user can speak again -================== -*/ -int G_FloodLimited( gentity_t *ent ) -{ - int deltatime, ms; - - if( g_floodMinTime.integer <= 0 ) - return 0; - - // handles !ent - if( G_admin_permission( ent, ADMF_NOCENSORFLOOD ) ) - return 0; - - deltatime = level.time - ent->client->pers.floodTime; - - ent->client->pers.floodDemerits += g_floodMinTime.integer - deltatime; - if( ent->client->pers.floodDemerits < 0 ) - ent->client->pers.floodDemerits = 0; - ent->client->pers.floodTime = level.time; - - ms = ent->client->pers.floodDemerits - g_floodMaxDemerits.integer; - if( ms <= 0 ) - return 0; - trap_SendServerCommand( ent - g_entities, va( "print \"You are flooding: " - "please wait %d second%s before trying again\n", - ( ms + 999 ) / 1000, ( ms > 1000 ) ? "s" : "" ) ); - return ms; -} - -commands_t cmds[ ] = { - { "a", CMD_MESSAGE|CMD_INTERMISSION, Cmd_AdminMessage_f }, - { "build", CMD_TEAM|CMD_LIVING, Cmd_Build_f }, - { "buy", CMD_HUMAN|CMD_LIVING, Cmd_Buy_f }, - { "callteamvote", CMD_MESSAGE|CMD_TEAM, Cmd_CallVote_f }, - { "callvote", CMD_MESSAGE, Cmd_CallVote_f }, - { "cb", 0, Cmd_Cb_f }, //NOTE: it's a command used only by cgame - { "class", CMD_TEAM, Cmd_Class_f }, - { "damage", CMD_CHEAT|CMD_LIVING, Cmd_Damage_f }, - { "deconstruct", CMD_TEAM|CMD_LIVING, Cmd_Destroy_f }, - { "destroy", CMD_CHEAT|CMD_TEAM|CMD_LIVING, Cmd_Destroy_f }, - { "follow", CMD_SPEC, Cmd_Follow_f }, - { "follownext", CMD_SPEC, Cmd_FollowCycle_f }, - { "followprev", CMD_SPEC, Cmd_FollowCycle_f }, - { "give", CMD_CHEAT|CMD_TEAM|CMD_LIVING, Cmd_Give_f }, - { "god", CMD_CHEAT|CMD_TEAM|CMD_LIVING, Cmd_God_f }, - { "ignore", 0, Cmd_Ignore_f }, - { "itemact", CMD_HUMAN|CMD_LIVING, Cmd_ActivateItem_f }, - { "itemdeact", CMD_HUMAN|CMD_LIVING, Cmd_DeActivateItem_f }, - { "itemtoggle", CMD_HUMAN|CMD_LIVING, Cmd_ToggleItem_f }, - { "kill", CMD_TEAM|CMD_LIVING, Cmd_Kill_f }, - { "levelshot", CMD_CHEAT, Cmd_LevelShot_f }, - { "listmaps", CMD_MESSAGE|CMD_INTERMISSION, Cmd_ListMaps_f }, - { "m", CMD_MESSAGE|CMD_INTERMISSION, Cmd_PrivateMessage_f }, - { "mt", CMD_MESSAGE|CMD_INTERMISSION, Cmd_PrivateMessage_f }, - { "noclip", CMD_CHEAT_TEAM, Cmd_Noclip_f }, - { "notarget", CMD_CHEAT|CMD_TEAM|CMD_LIVING, Cmd_Notarget_f }, - { "reload", CMD_HUMAN|CMD_LIVING, Cmd_Reload_f }, - { "say", CMD_MESSAGE|CMD_INTERMISSION, Cmd_Say_f }, - { "say_area", CMD_MESSAGE|CMD_TEAM|CMD_LIVING, Cmd_SayArea_f }, - { "say_team", CMD_MESSAGE|CMD_INTERMISSION, Cmd_Say_f }, - { "score", CMD_INTERMISSION, ScoreboardMessage }, - { "sell", CMD_HUMAN|CMD_LIVING, Cmd_Sell_f }, - { "setviewpos", CMD_CHEAT_TEAM, Cmd_SetViewpos_f }, - { "team", 0, Cmd_Team_f }, - { "teamvote", CMD_TEAM, Cmd_Vote_f }, - { "test", CMD_CHEAT, Cmd_Test_f }, - { "unignore", 0, Cmd_Ignore_f }, - { "vote", 0, Cmd_Vote_f }, - { "vsay", CMD_MESSAGE|CMD_INTERMISSION, Cmd_VSay_f }, - { "vsay_local", CMD_MESSAGE|CMD_INTERMISSION, Cmd_VSay_f }, - { "vsay_team", CMD_MESSAGE|CMD_INTERMISSION, Cmd_VSay_f }, - { "where", 0, Cmd_Where_f } -}; -static size_t numCmds = sizeof( cmds ) / sizeof( cmds[ 0 ] ); - -/* -================= -ClientCommand -================= -*/ -void ClientCommand( int clientNum ) -{ - gentity_t *ent; - char cmd[ MAX_TOKEN_CHARS ]; - commands_t *command; - - ent = g_entities + clientNum; - if( !ent->client || ent->client->pers.connected != CON_CONNECTED ) - return; // not fully in game yet - - trap_Argv( 0, cmd, sizeof( cmd ) ); - - command = bsearch( cmd, cmds, numCmds, sizeof( cmds[ 0 ] ), cmdcmp ); - - if( !command ) - { - if( !G_admin_cmd_check( ent ) ) - trap_SendServerCommand( clientNum, - va( "print \"Unknown command %s\n\"", cmd ) ); - return; - } - - // do tests here to reduce the amount of repeated code - - if( !( command->cmdFlags & CMD_INTERMISSION ) && - ( level.intermissiontime || level.pausedTime ) ) - return; - - if( command->cmdFlags & CMD_CHEAT && !g_cheats.integer ) - { - G_TriggerMenu( clientNum, MN_CMD_CHEAT ); - return; - } - - if( command->cmdFlags & CMD_MESSAGE && ( ent->client->pers.namelog->muted || - G_FloodLimited( ent ) ) ) - return; - - if( command->cmdFlags & CMD_TEAM && - ent->client->pers.teamSelection == TEAM_NONE ) - { - G_TriggerMenu( clientNum, MN_CMD_TEAM ); - return; - } - - if( command->cmdFlags & CMD_CHEAT_TEAM && !g_cheats.integer && - ent->client->pers.teamSelection != TEAM_NONE ) - { - G_TriggerMenu( clientNum, MN_CMD_CHEAT_TEAM ); - return; - } - - if( command->cmdFlags & CMD_SPEC && - ent->client->sess.spectatorState == SPECTATOR_NOT ) - { - G_TriggerMenu( clientNum, MN_CMD_SPEC ); - return; - } - - if( command->cmdFlags & CMD_ALIEN && - ent->client->pers.teamSelection != TEAM_ALIENS ) - { - G_TriggerMenu( clientNum, MN_CMD_ALIEN ); - return; - } - - if( command->cmdFlags & CMD_HUMAN && - ent->client->pers.teamSelection != TEAM_HUMANS ) - { - G_TriggerMenu( clientNum, MN_CMD_HUMAN ); - return; - } - - if( command->cmdFlags & CMD_LIVING && - ( ent->client->ps.stats[ STAT_HEALTH ] <= 0 || - ent->client->sess.spectatorState != SPECTATOR_NOT ) ) - { - G_TriggerMenu( clientNum, MN_CMD_LIVING ); - return; - } - - command->cmdHandler( ent ); -} - -void G_ListCommands( gentity_t *ent ) -{ - int i; - char out[ MAX_STRING_CHARS ] = ""; - int len, outlen; - - outlen = 0; - - for( i = 0; i < numCmds; i++ ) - { - len = strlen( cmds[ i ].cmdName ) + 1; - if( len + outlen >= sizeof( out ) - 1 ) - { - trap_SendServerCommand( ent - g_entities, va( "cmds%s\n", out ) ); - outlen = 0; - } - - strcpy( out + outlen, va( " %s", cmds[ i ].cmdName ) ); - outlen += len; - } - - trap_SendServerCommand( ent - g_entities, va( "cmds%s\n", out ) ); - G_admin_cmdlist( ent ); -} - -void G_DecolorString( char *in, char *out, int len ) -{ - qboolean decolor = qtrue; - - len--; - - while( *in && len > 0 ) { - if( *in == DECOLOR_OFF || *in == DECOLOR_ON ) - { - decolor = ( *in == DECOLOR_ON ); - in++; - continue; - } - if( Q_IsColorString( in ) && decolor ) { - in += 2; - continue; - } - *out++ = *in++; - len--; - } - *out = '\0'; -} - -void G_UnEscapeString( char *in, char *out, int len ) -{ - len--; - - while( *in && len > 0 ) - { - if( *in >= ' ' || *in == '\n' ) - { - *out++ = *in; - len--; - } - in++; - } - *out = '\0'; -} - -void Cmd_PrivateMessage_f( gentity_t *ent ) -{ - int pids[ MAX_CLIENTS ]; - char name[ MAX_NAME_LENGTH ]; - char cmd[ 12 ]; - char text[ MAX_STRING_CHARS ]; - char *msg; - char color; - int i, pcount; - int count = 0; - qboolean teamonly = qfalse; - char recipients[ MAX_STRING_CHARS ] = ""; - - if( !g_privateMessages.integer && ent ) - { - ADMP( "Sorry, but private messages have been disabled\n" ); - return; - } - - trap_Argv( 0, cmd, sizeof( cmd ) ); - if( trap_Argc( ) < 3 ) - { - ADMP( va( "usage: %s [name|slot#] [message]\n", cmd ) ); - return; - } - - if( !Q_stricmp( cmd, "mt" ) ) - teamonly = qtrue; - - trap_Argv( 1, name, sizeof( name ) ); - msg = ConcatArgs( 2 ); - pcount = G_ClientNumbersFromString( name, pids, MAX_CLIENTS ); - - G_CensorString( text, msg, sizeof( text ), ent ); - - // send the message - for( i = 0; i < pcount; i++ ) - { - if( G_SayTo( ent, &g_entities[ pids[ i ] ], - teamonly ? SAY_TPRIVMSG : SAY_PRIVMSG, text ) ) - { - count++; - Q_strcat( recipients, sizeof( recipients ), va( "%s" S_COLOR_WHITE ", ", - level.clients[ pids[ i ] ].pers.netname ) ); - } - } - - // report the results - color = teamonly ? COLOR_CYAN : COLOR_YELLOW; - - if( !count ) - ADMP( va( "^3No player matching ^7\'%s^7\' ^3to send message to.\n", - name ) ); - else - { - ADMP( va( "^%cPrivate message: ^7%s\n", color, text ) ); - // remove trailing ", " - recipients[ strlen( recipients ) - 2 ] = '\0'; - ADMP( va( "^%csent to %i player%s: " S_COLOR_WHITE "%s\n", color, count, - count == 1 ? "" : "s", recipients ) ); - - G_LogPrintf( "%s: %d \"%s" S_COLOR_WHITE "\" \"%s\": ^%c%s\n", - ( teamonly ) ? "TPrivMsg" : "PrivMsg", - ( ent ) ? ent - g_entities : -1, - ( ent ) ? ent->client->pers.netname : "console", - name, color, msg ); - } -} - -/* -================= -Cmd_AdminMessage_f - -Send a message to all active admins -================= -*/ -void Cmd_AdminMessage_f( gentity_t *ent ) -{ - // Check permissions and add the appropriate user [prefix] - if( !G_admin_permission( ent, ADMF_ADMINCHAT ) ) - { - if( !g_publicAdminMessages.integer ) - { - ADMP( "Sorry, but use of /a by non-admins has been disabled.\n" ); - return; - } - else - { - ADMP( "Your message has been sent to any available admins " - "and to the server logs.\n" ); - } - } - - if( trap_Argc( ) < 2 ) - { - ADMP( "usage: a [message]\n" ); - return; - } - - G_AdminMessage( ent, ConcatArgs( 1 ) ); -} - diff --git a/src/game/g_combat.c.orig b/src/game/g_combat.c.orig deleted file mode 100644 index f4e89ee..0000000 --- a/src/game/g_combat.c.orig +++ /dev/null @@ -1,1468 +0,0 @@ -/* -=========================================================================== -Copyright (C) 1999-2005 Id Software, Inc. -Copyright (C) 2000-2009 Darklegion Development - -This file is part of Tremulous. - -Tremulous 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. - -Tremulous is distributed in the hope that it will be -useful, but WITHOUT ANY WARRANTY; without even the implied warranty of -MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -GNU General Public License for more details. - -You should have received a copy of the GNU General Public License -along with Tremulous; if not, write to the Free Software -Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA -=========================================================================== -*/ - -#include "g_local.h" - -damageRegion_t g_damageRegions[ PCL_NUM_CLASSES ][ MAX_DAMAGE_REGIONS ]; -int g_numDamageRegions[ PCL_NUM_CLASSES ]; - -damageRegion_t g_armourRegions[ UP_NUM_UPGRADES ][ MAX_DAMAGE_REGIONS ]; -int g_numArmourRegions[ UP_NUM_UPGRADES ]; - -/* -============ -AddScore - -Adds score to the client -============ -*/ -void AddScore( gentity_t *ent, int score ) -{ - if( !ent->client ) - return; - - // make alien and human scores equivalent - if ( ent->client->pers.teamSelection == TEAM_ALIENS ) - { - score = rint( ((float)score) / 2.0f ); - } - - // scale values down to fit the scoreboard better - score = rint( ((float)score) / 50.0f ); - - ent->client->ps.persistant[ PERS_SCORE ] += score; - CalculateRanks( ); -} - -/* -================== -LookAtKiller -================== -*/ -void LookAtKiller( gentity_t *self, gentity_t *inflictor, gentity_t *attacker ) -{ - - if ( attacker && attacker != self ) - self->client->ps.stats[ STAT_VIEWLOCK ] = attacker - g_entities; - else if( inflictor && inflictor != self ) - self->client->ps.stats[ STAT_VIEWLOCK ] = inflictor - g_entities; - else - self->client->ps.stats[ STAT_VIEWLOCK ] = self - g_entities; -} - -// these are just for logging, the client prints its own messages -char *modNames[ ] = -{ - "MOD_UNKNOWN", - "MOD_SHOTGUN", - "MOD_BLASTER", - "MOD_PAINSAW", - "MOD_MACHINEGUN", - "MOD_CHAINGUN", - "MOD_PRIFLE", - "MOD_MDRIVER", - "MOD_LASGUN", - "MOD_LCANNON", - "MOD_LCANNON_SPLASH", - "MOD_FLAMER", - "MOD_FLAMER_SPLASH", - "MOD_GRENADE", - "MOD_WATER", - "MOD_SLIME", - "MOD_LAVA", - "MOD_CRUSH", - "MOD_TELEFRAG", - "MOD_FALLING", - "MOD_SUICIDE", - "MOD_TARGET_LASER", - "MOD_TRIGGER_HURT", - - "MOD_ABUILDER_CLAW", - "MOD_LEVEL0_BITE", - "MOD_LEVEL1_CLAW", - "MOD_LEVEL1_PCLOUD", - "MOD_LEVEL3_CLAW", - "MOD_LEVEL3_POUNCE", - "MOD_LEVEL3_BOUNCEBALL", - "MOD_LEVEL2_CLAW", - "MOD_LEVEL2_ZAP", - "MOD_LEVEL4_CLAW", - "MOD_LEVEL4_TRAMPLE", - "MOD_LEVEL4_CRUSH", - - "MOD_SLOWBLOB", - "MOD_POISON", - "MOD_SWARM", - - "MOD_HSPAWN", - "MOD_TESLAGEN", - "MOD_MGTURRET", - "MOD_REACTOR", - - "MOD_ASPAWN", - "MOD_ATUBE", - "MOD_OVERMIND", - "MOD_DECONSTRUCT", - "MOD_REPLACE", - "MOD_NOCREEP" -}; - -/* -================== -G_RewardAttackers - -Function to distribute rewards to entities that killed this one. -Returns the total damage dealt. -================== -*/ -float G_RewardAttackers( gentity_t *self ) -{ - float value, totalDamage = 0; - int team, i, maxHealth = 0; - int alienCredits = 0, humanCredits = 0; - gentity_t *player; - - // Total up all the damage done by non-teammates - for( i = 0; i < level.maxclients; i++ ) - { - player = g_entities + i; - - if( !OnSameTeam( self, player ) || - self->buildableTeam != player->client->ps.stats[ STAT_TEAM ] ) - totalDamage += (float)self->credits[ i ]; - } - - if( totalDamage <= 0.0f ) - return 0.0f; - - // Only give credits for killing players and buildables - if( self->client ) - { - value = BG_GetValueOfPlayer( &self->client->ps ); - team = self->client->pers.teamSelection; - maxHealth = self->client->ps.stats[ STAT_MAX_HEALTH ]; - } - else if( self->s.eType == ET_BUILDABLE ) - { - value = BG_Buildable( self->s.modelindex, self->cuboidSize )->value; - - // only give partial credits for a buildable not yet completed - if( !self->spawned ) - { - value *= (float)( level.time - self->buildTime ) / - BG_Buildable( self->s.modelindex, self->cuboidSize )->buildTime; - } - - team = self->buildableTeam; - maxHealth = BG_Buildable( self->s.modelindex, self->cuboidSize )->health; - } - else - return totalDamage; - - // Give credits and empty the array - for( i = 0; i < level.maxclients; i++ ) - { - int stageValue = value * self->credits[ i ] / totalDamage; - player = g_entities + i; - - if( player->client->pers.teamSelection != team ) - { - if( totalDamage < maxHealth ) - stageValue *= totalDamage / maxHealth; - - if( !self->credits[ i ] || player->client->ps.stats[ STAT_TEAM ] == team ) - continue; - - AddScore( player, stageValue ); - - // killing buildables earns score, but not credits - if( self->s.eType != ET_BUILDABLE ) - { - G_AddCreditToClient( player->client, stageValue, qtrue ); - - // add to stage counters - if( player->client->ps.stats[ STAT_TEAM ] == TEAM_ALIENS ) - alienCredits += stageValue; - else if( player->client->ps.stats[ STAT_TEAM ] == TEAM_HUMANS ) - humanCredits += stageValue; - } - } - self->credits[ i ] = 0; - } - - if( alienCredits ) - { - trap_Cvar_Set( "g_alienCredits", - va( "%d", g_alienCredits.integer + alienCredits ) ); - trap_Cvar_Update( &g_alienCredits ); - } - if( humanCredits ) - { - trap_Cvar_Set( "g_humanCredits", - va( "%d", g_humanCredits.integer + humanCredits ) ); - trap_Cvar_Update( &g_humanCredits ); - } - - return totalDamage; -} - -/* -================== -player_die -================== -*/ -void player_die( gentity_t *self, gentity_t *inflictor, gentity_t *attacker, int damage, int meansOfDeath ) -{ - gentity_t *ent, *ent2; - int anim; - int killer; - int i; - char *killerName, *obit; - vec3_t dir; - - if( self->client->ps.pm_type == PM_DEAD ) - return; - - if( level.intermissiontime ) - return; - - self->client->ps.pm_type = PM_DEAD; - self->suicideTime = 0; - - if( attacker ) - { - killer = attacker->s.number; - - if( attacker->client ) - killerName = attacker->client->pers.netname; - else - killerName = "<world>"; - } - else - { - killer = ENTITYNUM_WORLD; - killerName = "<world>"; - } - - if( meansOfDeath < 0 || meansOfDeath >= sizeof( modNames ) / sizeof( modNames[0] ) ) - // fall back on the number - obit = va( "%d", meansOfDeath ); - else - obit = modNames[ meansOfDeath ]; - - G_LogPrintf( "Die: %d %d %s: %s" S_COLOR_WHITE " killed %s\n", - killer, - self - g_entities, - obit, - killerName, - self->client->pers.netname ); - - // deactivate all upgrades - for( i = UP_NONE + 1; i < UP_NUM_UPGRADES; i++ ) - BG_DeactivateUpgrade( i, self->client->ps.stats ); - - // kill all player's buildables if they havent spawned yet - // this should eliminate build timer hacks for ever - dir[0] = dir[1] = 0.0f; - dir[2] = 1.0f; - - for( i = MAX_CLIENTS, ent = g_entities + i; i < level.num_entities; i++, ent++ ) - { - if( ent->s.eType != ET_BUILDABLE ) - continue; - - if( ent == self ) - continue; - - if( ent->spawned ) - continue; - - if( ent->builtBy != self->client->ps.clientNum ) - continue; - - G_Damage( ent, self, attacker, dir, dir, ent->health, 0, MOD_DECONSTRUCT ); - } - - // broadcast the death event to everyone - ent = G_TempEntity( self->r.currentOrigin, EV_OBITUARY ); - ent->s.eventParm = meansOfDeath; - ent->s.otherEntityNum = self->s.number; - ent->s.otherEntityNum2 = killer; - ent->r.svFlags = SVF_BROADCAST; // send to everyone - - self->enemy = attacker; - self->client->ps.persistant[ PERS_KILLED ]++; - - if( attacker && attacker->client ) - { - attacker->client->lastkilled_client = self->s.number; - - if( ( attacker == self || OnSameTeam( self, attacker ) ) && meansOfDeath != MOD_HSPAWN ) - { - //punish team kills and suicides - if( attacker->client->ps.stats[ STAT_TEAM ] == TEAM_ALIENS ) - { - G_AddCreditToClient( attacker->client, -ALIEN_TK_SUICIDE_PENALTY, qtrue ); - AddScore( attacker, -ALIEN_TK_SUICIDE_PENALTY ); - } - else if( attacker->client->ps.stats[ STAT_TEAM ] == TEAM_HUMANS ) - { - G_AddCreditToClient( attacker->client, -HUMAN_TK_SUICIDE_PENALTY, qtrue ); - AddScore( attacker, -HUMAN_TK_SUICIDE_PENALTY ); - } - } - } - else if( attacker->s.eType != ET_BUILDABLE ) - { - if( self->client->ps.stats[ STAT_TEAM ] == TEAM_ALIENS ) - AddScore( self, -ALIEN_TK_SUICIDE_PENALTY ); - else if( self->client->ps.stats[ STAT_TEAM ] == TEAM_HUMANS ) - AddScore( self, -HUMAN_TK_SUICIDE_PENALTY ); - } - - // give credits for killing this player - G_RewardAttackers( self ); - - ScoreboardMessage( self ); // show scores - - // send updated scores to any clients that are following this one, - // or they would get stale scoreboards - for( i = 0 ; i < level.maxclients ; i++ ) - { - gclient_t *client; - - client = &level.clients[ i ]; - if( client->pers.connected != CON_CONNECTED ) - continue; - - if( client->sess.spectatorState == SPECTATOR_NOT ) - continue; - - if( client->sess.spectatorClient == self->s.number ) - ScoreboardMessage( g_entities + i ); - } - - VectorCopy( self->s.origin, self->client->pers.lastDeathLocation ); - - self->takedamage = qfalse; // can still be gibbed - - self->s.weapon = WP_NONE; - self->r.contents = CONTENTS_CORPSE; - - self->s.angles[ PITCH ] = 0; - self->s.angles[ ROLL ] = 0; - self->s.angles[ YAW ] = self->s.apos.trBase[ YAW ]; - LookAtKiller( self, inflictor, attacker ); - - VectorCopy( self->s.angles, self->client->ps.viewangles ); - - self->s.loopSound = 0; - - self->r.maxs[ 2 ] = -8; - - // don't allow respawn until the death anim is done - // g_forcerespawn may force spawning at some later time - self->client->respawnTime = level.time + 1700; - - // clear misc - memset( self->client->ps.misc, 0, sizeof( self->client->ps.misc ) ); - - { - // normal death - static int i; - - if( !( self->client->ps.persistant[ PERS_STATE ] & PS_NONSEGMODEL ) ) - { - switch( i ) - { - case 0: - anim = BOTH_DEATH1; - break; - case 1: - anim = BOTH_DEATH2; - break; - case 2: - default: - anim = BOTH_DEATH3; - break; - } - } - else - { - switch( i ) - { - case 0: - anim = NSPA_DEATH1; - break; - case 1: - anim = NSPA_DEATH2; - break; - case 2: - default: - anim = NSPA_DEATH3; - break; - } - } - - self->client->ps.legsAnim = - ( ( self->client->ps.legsAnim & ANIM_TOGGLEBIT ) ^ ANIM_TOGGLEBIT ) | anim; - - if( !( self->client->ps.persistant[ PERS_STATE ] & PS_NONSEGMODEL ) ) - { - self->client->ps.torsoAnim = - ( ( self->client->ps.torsoAnim & ANIM_TOGGLEBIT ) ^ ANIM_TOGGLEBIT ) | anim; - } - - // use own entityid if killed by non-client to prevent uint8_t overflow - G_AddEvent( self, EV_DEATH1 + i, - ( killer < MAX_CLIENTS ) ? killer : self - g_entities ); - - // globally cycle through the different death animations - i = ( i + 1 ) % 3; - } - - trap_LinkEntity( self ); -} - -/* -=============== -G_ParseDmgScript -=============== -*/ -static int G_ParseDmgScript( damageRegion_t *regions, char *buf ) -{ - char *token; - float angleSpan, heightSpan; - int count; - - for( count = 0; ; count++ ) - { - token = COM_Parse( &buf ); - if( !token[ 0 ] ) - break; - - if( strcmp( token, "{" ) ) - { - COM_ParseError( "Missing {" ); - break; - } - - if( count >= MAX_DAMAGE_REGIONS ) - { - COM_ParseError( "Max damage regions exceeded" ); - break; - } - - // defaults - regions[ count ].name[ 0 ] = '\0'; - regions[ count ].minHeight = 0.0f; - regions[ count ].maxHeight = 1.0f; - regions[ count ].minAngle = 0.0f; - regions[ count ].maxAngle = 360.0f; - regions[ count ].modifier = 1.0f; - regions[ count ].crouch = qfalse; - - while( 1 ) - { - token = COM_ParseExt( &buf, qtrue ); - - if( !token[ 0 ] ) - { - COM_ParseError( "Unexpected end of file" ); - break; - } - - if( !Q_stricmp( token, "}" ) ) - { - break; - } - else if( !strcmp( token, "name" ) ) - { - token = COM_ParseExt( &buf, qfalse ); - if( token[ 0 ] ) - Q_strncpyz( regions[ count ].name, token, - sizeof( regions[ count ].name ) ); - } - else if( !strcmp( token, "minHeight" ) ) - { - token = COM_ParseExt( &buf, qfalse ); - if( !token[ 0 ] ) - strcpy( token, "0" ); - regions[ count ].minHeight = atof( token ); - } - else if( !strcmp( token, "maxHeight" ) ) - { - token = COM_ParseExt( &buf, qfalse ); - if( !token[ 0 ] ) - strcpy( token, "100" ); - regions[ count ].maxHeight = atof( token ); - } - else if( !strcmp( token, "minAngle" ) ) - { - token = COM_ParseExt( &buf, qfalse ); - if( !token[ 0 ] ) - strcpy( token, "0" ); - regions[ count ].minAngle = atoi( token ); - } - else if( !strcmp( token, "maxAngle" ) ) - { - token = COM_ParseExt( &buf, qfalse ); - if( !token[ 0 ] ) - strcpy( token, "360" ); - regions[ count ].maxAngle = atoi( token ); - } - else if( !strcmp( token, "modifier" ) ) - { - token = COM_ParseExt( &buf, qfalse ); - if( !token[ 0 ] ) - strcpy( token, "1.0" ); - regions[ count ].modifier = atof( token ); - } - else if( !strcmp( token, "crouch" ) ) - { - regions[ count ].crouch = qtrue; - } - else - { - COM_ParseWarning("Unknown token \"%s\"", token); - } - } - - // Angle portion covered - angleSpan = regions[ count ].maxAngle - regions[ count ].minAngle; - if( angleSpan < 0.0f ) - angleSpan += 360.0f; - angleSpan /= 360.0f; - - // Height portion covered - heightSpan = regions[ count ].maxHeight - regions[ count ].minHeight; - if( heightSpan < 0.0f ) - heightSpan = -heightSpan; - if( heightSpan > 1.0f ) - heightSpan = 1.0f; - - regions[ count ].area = angleSpan * heightSpan; - if( !regions[ count ].area ) - regions[ count ].area = 0.00001f; - } - - return count; -} - -/* -============ -GetRegionDamageModifier -============ -*/ -static float GetRegionDamageModifier( gentity_t *targ, int class, int piece ) -{ - damageRegion_t *regions, *overlap; - float modifier = 0.0f, areaSum = 0.0f; - int j, i; - qboolean crouch; - - crouch = targ->client->ps.pm_flags & PMF_DUCKED; - overlap = &g_damageRegions[ class ][ piece ]; - - if( g_debugDamage.integer > 2 ) - G_Printf( "GetRegionDamageModifier():\n" - ". bodyRegion = [%d %d %f %f] (%s)\n" - ". modifier = %f\n", - overlap->minAngle, overlap->maxAngle, - overlap->minHeight, overlap->maxHeight, - overlap->name, overlap->modifier ); - - // Find the armour layer modifier, assuming that none of the armour regions - // overlap and that any areas that are not covered have a modifier of 1.0 - for( j = UP_NONE + 1; j < UP_NUM_UPGRADES; j++ ) - { - if( !BG_InventoryContainsUpgrade( j, targ->client->ps.stats ) || - !g_numArmourRegions[ j ] ) - continue; - regions = g_armourRegions[ j ]; - - for( i = 0; i < g_numArmourRegions[ j ]; i++ ) - { - float overlapMaxA, regionMinA, regionMaxA, angleSpan, heightSpan, area; - - if( regions[ i ].crouch != crouch ) - continue; - - // Convert overlap angle to 0 to max - overlapMaxA = overlap->maxAngle - overlap->minAngle; - if( overlapMaxA < 0.0f ) - overlapMaxA += 360.0f; - - // Convert region angles to match overlap - regionMinA = regions[ i ].minAngle - overlap->minAngle; - if( regionMinA < 0.0f ) - regionMinA += 360.0f; - regionMaxA = regions[ i ].maxAngle - overlap->minAngle; - if( regionMaxA < 0.0f ) - regionMaxA += 360.0f; - - // Overlapping Angle portion - if( regionMinA <= regionMaxA ) - { - angleSpan = 0.0f; - if( regionMinA < overlapMaxA ) - { - if( regionMaxA > overlapMaxA ) - regionMaxA = overlapMaxA; - angleSpan = regionMaxA - regionMinA; - } - } - else - { - if( regionMaxA > overlapMaxA ) - regionMaxA = overlapMaxA; - angleSpan = regionMaxA; - if( regionMinA < overlapMaxA ) - angleSpan += overlapMaxA - regionMinA; - } - angleSpan /= 360.0f; - - // Overlapping height portion - heightSpan = MIN( overlap->maxHeight, regions[ i ].maxHeight ) - - MAX( overlap->minHeight, regions[ i ].minHeight ); - if( heightSpan < 0.0f ) - heightSpan = 0.0f; - if( heightSpan > 1.0f ) - heightSpan = 1.0f; - - if( g_debugDamage.integer > 2 ) - G_Printf( ". armourRegion = [%d %d %f %f] (%s)\n" - ". . modifier = %f\n" - ". . angleSpan = %f\n" - ". . heightSpan = %f\n", - regions[ i ].minAngle, regions[ i ].maxAngle, - regions[ i ].minHeight, regions[ i ].maxHeight, - regions[ i ].name, regions[ i ].modifier, - angleSpan, heightSpan ); - - areaSum += area = angleSpan * heightSpan; - modifier += regions[ i ].modifier * area; - } - } - - if( g_debugDamage.integer > 2 ) - G_Printf( ". areaSum = %f\n" - ". armourModifier = %f\n", areaSum, modifier ); - - return overlap->modifier * ( overlap->area + modifier - areaSum ); -} - -/* -============ -GetNonLocDamageModifier -============ -*/ -static float GetNonLocDamageModifier( gentity_t *targ, int class ) -{ - float modifier = 0.0f, area = 0.0f, scale = 0.0f; - int i; - qboolean crouch; - - // For every body region, use stretch-armor formula to apply armour modifier - // for any overlapping area that armour shares with the body region - crouch = targ->client->ps.pm_flags & PMF_DUCKED; - for( i = 0; i < g_numDamageRegions[ class ]; i++ ) - { - damageRegion_t *region; - - region = &g_damageRegions[ class ][ i ]; - - if( region->crouch != crouch ) - continue; - - modifier += GetRegionDamageModifier( targ, class, i ); - - scale += region->modifier * region->area; - area += region->area; - - } - - modifier = !scale ? 1.0f : 1.0f + ( modifier / scale - 1.0f ) * area; - - if( g_debugDamage.integer > 1 ) - G_Printf( "GetNonLocDamageModifier() modifier:%f, area:%f, scale:%f\n", - modifier, area, scale ); - - return modifier; -} - -/* -============ -GetPointDamageModifier - -Returns the damage region given an angle and a height proportion -============ -*/ -static float GetPointDamageModifier( gentity_t *targ, damageRegion_t *regions, - int len, float angle, float height ) -{ - float modifier = 1.0f; - int i; - - for( i = 0; i < len; i++ ) - { - if( regions[ i ].crouch != ( targ->client->ps.pm_flags & PMF_DUCKED ) ) - continue; - - // Angle must be within range - if( ( regions[ i ].minAngle <= regions[ i ].maxAngle && - ( angle < regions[ i ].minAngle || - angle > regions[ i ].maxAngle ) ) || - ( regions[ i ].minAngle > regions[ i ].maxAngle && - angle > regions[ i ].maxAngle && angle < regions[ i ].minAngle ) ) - continue; - - // Height must be within range - if( height < regions[ i ].minHeight || height > regions[ i ].maxHeight ) - continue; - - modifier *= regions[ i ].modifier; - } - - if( g_debugDamage.integer ) - G_Printf( "GetDamageRegionModifier(angle = %f, height = %f): %f\n", - angle, height, modifier ); - - return modifier; -} - -/* -============ -G_CalcDamageModifier -============ -*/ -static float G_CalcDamageModifier( vec3_t point, gentity_t *targ, gentity_t *attacker, int class, int dflags ) -{ - vec3_t targOrigin, bulletPath, bulletAngle, pMINUSfloor, floor, normal; - float clientHeight, hitRelative, hitRatio, modifier; - int hitRotation, i; - - if( point == NULL ) - return 1.0f; - - // Don't need to calculate angles and height for non-locational damage - if( dflags & DAMAGE_NO_LOCDAMAGE ) - return GetNonLocDamageModifier( targ, class ); - - // Get the point location relative to the floor under the target - if( g_unlagged.integer && targ->client && targ->client->unlaggedCalc.used ) - VectorCopy( targ->client->unlaggedCalc.origin, targOrigin ); - else - VectorCopy( targ->r.currentOrigin, targOrigin ); - - BG_GetClientNormal( &targ->client->ps, normal ); - VectorMA( targOrigin, targ->r.mins[ 2 ], normal, floor ); - VectorSubtract( point, floor, pMINUSfloor ); - - // Get the proportion of the target height where the hit landed - clientHeight = targ->r.maxs[ 2 ] - targ->r.mins[ 2 ]; - - if( !clientHeight ) - clientHeight = 1.0f; - - hitRelative = DotProduct( normal, pMINUSfloor ) / VectorLength( normal ); - - if( hitRelative < 0.0f ) - hitRelative = 0.0f; - - if( hitRelative > clientHeight ) - hitRelative = clientHeight; - - hitRatio = hitRelative / clientHeight; - - // Get the yaw of the attack relative to the target's view yaw - VectorSubtract( point, targOrigin, bulletPath ); - vectoangles( bulletPath, bulletAngle ); - - hitRotation = AngleNormalize360( targ->client->ps.viewangles[ YAW ] - - bulletAngle[ YAW ] ); - - // Get modifiers from the target's damage regions - modifier = GetPointDamageModifier( targ, g_damageRegions[ class ], - g_numDamageRegions[ class ], - hitRotation, hitRatio ); - - for( i = UP_NONE + 1; i < UP_NUM_UPGRADES; i++ ) - { - if( BG_InventoryContainsUpgrade( i, targ->client->ps.stats ) ) - { - modifier *= GetPointDamageModifier( targ, g_armourRegions[ i ], - g_numArmourRegions[ i ], - hitRotation, hitRatio ); - } - } - - return modifier; -} - - -/* -============ -G_InitDamageLocations -============ -*/ -void G_InitDamageLocations( void ) -{ - char *modelName; - char filename[ MAX_QPATH ]; - int i; - int len; - fileHandle_t fileHandle; - char buffer[ MAX_DAMAGE_REGION_TEXT ]; - - for( i = PCL_NONE + 1; i < PCL_NUM_CLASSES; i++ ) - { - modelName = BG_ClassConfig( i )->modelName; - Com_sprintf( filename, sizeof( filename ), - "models/players/%s/locdamage.cfg", modelName ); - - len = trap_FS_FOpenFile( filename, &fileHandle, FS_READ ); - if ( !fileHandle ) - { - G_Printf( S_COLOR_RED "file not found: %s\n", filename ); - continue; - } - - if( len >= MAX_DAMAGE_REGION_TEXT ) - { - G_Printf( S_COLOR_RED "file too large: %s is %i, max allowed is %i", - filename, len, MAX_DAMAGE_REGION_TEXT ); - trap_FS_FCloseFile( fileHandle ); - continue; - } - - COM_BeginParseSession( filename ); - - trap_FS_Read( buffer, len, fileHandle ); - buffer[len] = 0; - trap_FS_FCloseFile( fileHandle ); - - g_numDamageRegions[ i ] = G_ParseDmgScript( g_damageRegions[ i ], buffer ); - } - - for( i = UP_NONE + 1; i < UP_NUM_UPGRADES; i++ ) - { - modelName = BG_Upgrade( i )->name; - Com_sprintf( filename, sizeof( filename ), "armour/%s.armour", modelName ); - - len = trap_FS_FOpenFile( filename, &fileHandle, FS_READ ); - - //no file - no parsage - if ( !fileHandle ) - continue; - - if( len >= MAX_DAMAGE_REGION_TEXT ) - { - G_Printf( S_COLOR_RED "file too large: %s is %i, max allowed is %i", - filename, len, MAX_DAMAGE_REGION_TEXT ); - trap_FS_FCloseFile( fileHandle ); - continue; - } - - COM_BeginParseSession( filename ); - - trap_FS_Read( buffer, len, fileHandle ); - buffer[len] = 0; - trap_FS_FCloseFile( fileHandle ); - - g_numArmourRegions[ i ] = G_ParseDmgScript( g_armourRegions[ i ], buffer ); - } -} - - -/* -============ -T_Damage - -targ entity that is being damaged -inflictor entity that is causing the damage -attacker entity that caused the inflictor to damage targ - example: targ=monster, inflictor=rocket, attacker=player - -dir direction of the attack for knockback -point point at which the damage is being inflicted, used for headshots -damage amount of damage being inflicted -knockback force to be applied against targ as a result of the damage - -inflictor, attacker, dir, and point can be NULL for environmental effects - -dflags these flags are used to control how T_Damage works - DAMAGE_RADIUS damage was indirect (from a nearby explosion) - DAMAGE_NO_ARMOR armor does not protect from this damage - DAMAGE_NO_KNOCKBACK do not affect velocity, just view angles - DAMAGE_NO_PROTECTION kills godmode, armor, everything -============ -*/ - -// team is the team that is immune to this damage -void G_SelectiveDamage( gentity_t *targ, gentity_t *inflictor, gentity_t *attacker, - vec3_t dir, vec3_t point, int damage, int dflags, int mod, int team ) -{ - if( targ->client && ( team != targ->client->ps.stats[ STAT_TEAM ] ) ) - G_Damage( targ, inflictor, attacker, dir, point, damage, dflags, mod ); -} - -void G_Damage( gentity_t *targ, gentity_t *inflictor, gentity_t *attacker, - vec3_t dir, vec3_t point, int damage, int dflags, int mod ) -{ - gclient_t *client; - int take; - int asave = 0; - int knockback; - - // Can't deal damage sometimes - if( !targ->takedamage || targ->health <= 0 || level.intermissionQueued ) - return; - - if( !inflictor ) - inflictor = &g_entities[ ENTITYNUM_WORLD ]; - - if( !attacker ) - attacker = &g_entities[ ENTITYNUM_WORLD ]; - - // shootable doors / buttons don't actually have any health - if( targ->s.eType == ET_MOVER ) - { - if( targ->use && ( targ->moverState == MOVER_POS1 || - targ->moverState == ROTATOR_POS1 ) ) - targ->use( targ, inflictor, attacker ); - - return; - } - - client = targ->client; - if( client && client->noclip ) - return; - - if( !dir ) - dflags |= DAMAGE_NO_KNOCKBACK; - else - VectorNormalize( dir ); - - knockback = damage; - - if( inflictor->s.weapon != WP_NONE ) - { - knockback = (int)( (float)knockback * - BG_Weapon( inflictor->s.weapon )->knockbackScale ); - } - - if( targ->client ) - { - knockback = (int)( (float)knockback * - BG_Class( targ->client->ps.stats[ STAT_CLASS ] )->knockbackScale ); - } - - // Too much knockback from falling really far makes you "bounce" and - // looks silly. However, none at all also looks bad. Cap it. - if( mod == MOD_FALLING && knockback > 50 ) - knockback = 50; - - if( knockback > 200 ) - knockback = 200; - - if( targ->flags & FL_NO_KNOCKBACK ) - knockback = 0; - - if( dflags & DAMAGE_NO_KNOCKBACK ) - knockback = 0; - - // figure momentum add, even if the damage won't be taken - if( knockback && targ->client ) - { - vec3_t kvel; - float mass; - - mass = 200; - - VectorScale( dir, g_knockback.value * (float)knockback / mass, kvel ); - VectorAdd( targ->client->ps.velocity, kvel, targ->client->ps.velocity ); - - // set the timer so that the other client can't cancel - // out the movement immediately - if( !targ->client->ps.pm_time ) - { - int t; - - t = knockback * 2; - if( t < 50 ) - t = 50; - - if( t > 200 ) - t = 200; - - targ->client->ps.pm_time = t; - targ->client->ps.pm_flags |= PMF_TIME_KNOCKBACK; - } - } - - // don't do friendly fire on movement attacks - if( ( mod == MOD_LEVEL4_TRAMPLE || mod == MOD_LEVEL3_POUNCE || - mod == MOD_LEVEL4_CRUSH ) && - targ->s.eType == ET_BUILDABLE && targ->buildableTeam == TEAM_ALIENS ) - { - return; - } - - // check for completely getting out of the damage - if( !( dflags & DAMAGE_NO_PROTECTION ) ) - { - - // if TF_NO_FRIENDLY_FIRE is set, don't do damage to the target - // if the attacker was on the same team - if( targ != attacker && OnSameTeam( targ, attacker ) ) - { - // don't do friendly fire on movement attacks - if( mod == MOD_LEVEL4_TRAMPLE || mod == MOD_LEVEL3_POUNCE || - mod == MOD_LEVEL4_CRUSH ) - return; - - // if dretchpunt is enabled and this is a dretch, do dretchpunt instead of damage - if( g_dretchPunt.integer && - targ->client->ps.stats[ STAT_CLASS ] == PCL_ALIEN_LEVEL0 ) - { - vec3_t dir, push; - - VectorSubtract( targ->r.currentOrigin, attacker->r.currentOrigin, dir ); - VectorNormalizeFast( dir ); - VectorScale( dir, ( damage * 10.0f ), push ); - push[2] = 64.0f; - VectorAdd( targ->client->ps.velocity, push, targ->client->ps.velocity ); - return; - } - - // check if friendly fire has been disabled - if( !g_friendlyFire.integer ) - { - return; - } - } - - if( targ->s.eType == ET_BUILDABLE && attacker->client && - mod != MOD_DECONSTRUCT && mod != MOD_SUICIDE && - mod != MOD_REPLACE && mod != MOD_NOCREEP ) - { - if( targ->buildableTeam == attacker->client->pers.teamSelection && - !g_friendlyBuildableFire.integer ) - { - return; - } - - // base is under attack warning if DCC'd - if( targ->buildableTeam == TEAM_HUMANS && G_FindDCC( targ ) && - level.time > level.humanBaseAttackTimer ) - { - level.humanBaseAttackTimer = level.time + DC_ATTACK_PERIOD; - G_BroadcastEvent( EV_DCC_ATTACK, 0 ); - } - } - - // check for godmode - if ( targ->flags & FL_GODMODE ) - return; - } - - // add to the attacker's hit counter - if( attacker->client && targ != attacker && targ->health > 0 - && targ->s.eType != ET_MISSILE - && targ->s.eType != ET_GENERAL ) - { - if( OnSameTeam( targ, attacker ) ) - attacker->client->ps.persistant[ PERS_HITS ]--; - else - attacker->client->ps.persistant[ PERS_HITS ]++; - } - - take = damage; - - // add to the damage inflicted on a player this frame - // the total will be turned into screen blends and view angle kicks - // at the end of the frame - if( client ) - { - if( attacker ) - client->ps.persistant[ PERS_ATTACKER ] = attacker->s.number; - else - client->ps.persistant[ PERS_ATTACKER ] = ENTITYNUM_WORLD; - - client->damage_armor += asave; - client->damage_blood += take; - client->damage_knockback += knockback; - - if( dir ) - { - VectorCopy ( dir, client->damage_from ); - client->damage_fromWorld = qfalse; - } - else - { - VectorCopy ( targ->r.currentOrigin, client->damage_from ); - client->damage_fromWorld = qtrue; - } - - // set the last client who damaged the target - targ->client->lasthurt_client = attacker->s.number; - targ->client->lasthurt_mod = mod; - take = (int)( take * G_CalcDamageModifier( point, targ, attacker, - client->ps.stats[ STAT_CLASS ], - dflags ) + 0.5f ); - - //if boosted poison every attack - if( attacker->client && attacker->client->ps.stats[ STAT_STATE ] & SS_BOOSTED ) - { - if( (targ->client->ps.stats[ STAT_TEAM ] == TEAM_HUMANS ) && - mod != MOD_LEVEL2_ZAP && mod != MOD_POISON && - mod != MOD_LEVEL1_PCLOUD && mod != MOD_HSPAWN && - mod != MOD_ASPAWN && targ->client->poisonImmunityTime < level.time ) - { - targ->client->ps.stats[ STAT_STATE ] |= SS_POISONED; - targ->client->lastPoisonTime = level.time; - targ->client->lastPoisonClient = attacker; - } - } - } - - if( take < 1 ) - take = 1; - - if( g_debugDamage.integer ) - { - G_Printf( "%i: client:%i health:%i damage:%i armor:%i\n", level.time, targ->s.number, - targ->health, take, asave ); - } - - // do the damage - if( take ) - { - targ->health = targ->health - take; - - if( targ->client ) - targ->client->ps.stats[ STAT_HEALTH ] = targ->health; - - targ->lastDamageTime = level.time; - targ->nextRegenTime = level.time + ALIEN_REGEN_DAMAGE_TIME; - - // add to the attackers "account" on the target - if( attacker->client && attacker != targ ) - targ->credits[ attacker->client->ps.clientNum ] += take; - - if( targ->health <= 0 ) - { - if( client ) - targ->flags |= FL_NO_KNOCKBACK; - - if( targ->health < -999 ) - targ->health = -999; - - targ->enemy = attacker; - targ->die( targ, inflictor, attacker, take, mod ); - return; - } - else if( targ->pain ) - { - //if(targ->s.eType==ET_BUILDABLE&&BG_Buildable(targ->s.modelindex,NULL)->cuboid) - // VectorCopy(point,targ->s.angles2); - targ->pain( targ, attacker, take ); - } - } -} - - -/* -============ -CanDamage - -Returns qtrue if the inflictor can directly damage the target. Used for -explosions and melee attacks. -============ -*/ -qboolean CanDamage( gentity_t *targ, vec3_t origin ) -{ - vec3_t dest; - trace_t tr; - vec3_t midpoint; - - // use the midpoint of the bounds instead of the origin, because - // bmodels may have their origin is 0,0,0 - VectorAdd( targ->r.absmin, targ->r.absmax, midpoint ); - VectorScale( midpoint, 0.5, midpoint ); - - VectorCopy( midpoint, dest ); - trap_Trace( &tr, origin, vec3_origin, vec3_origin, dest, ENTITYNUM_NONE, MASK_SOLID ); - if( tr.fraction == 1.0 || tr.entityNum == targ->s.number ) - return qtrue; - - // this should probably check in the plane of projection, - // rather than in world coordinate, and also include Z - VectorCopy( midpoint, dest ); - dest[ 0 ] += 15.0; - dest[ 1 ] += 15.0; - trap_Trace( &tr, origin, vec3_origin, vec3_origin, dest, ENTITYNUM_NONE, MASK_SOLID ); - if( tr.fraction == 1.0 ) - return qtrue; - - VectorCopy( midpoint, dest ); - dest[ 0 ] += 15.0; - dest[ 1 ] -= 15.0; - trap_Trace( &tr, origin, vec3_origin, vec3_origin, dest, ENTITYNUM_NONE, MASK_SOLID ); - if( tr.fraction == 1.0 ) - return qtrue; - - VectorCopy( midpoint, dest ); - dest[ 0 ] -= 15.0; - dest[ 1 ] += 15.0; - trap_Trace( &tr, origin, vec3_origin, vec3_origin, dest, ENTITYNUM_NONE, MASK_SOLID ); - if( tr.fraction == 1.0 ) - return qtrue; - - VectorCopy( midpoint, dest ); - dest[ 0 ] -= 15.0; - dest[ 1 ] -= 15.0; - trap_Trace( &tr, origin, vec3_origin, vec3_origin, dest, ENTITYNUM_NONE, MASK_SOLID ); - if( tr.fraction == 1.0 ) - return qtrue; - - return qfalse; -} - -/* -============ -G_SelectiveRadiusDamage -============ -*/ -qboolean G_SelectiveRadiusDamage( vec3_t origin, gentity_t *attacker, float damage, - float radius, gentity_t *ignore, int mod, int team ) -{ - float points, dist; - gentity_t *ent; - int entityList[ MAX_GENTITIES ]; - int numListedEntities; - vec3_t mins, maxs; - vec3_t v; - vec3_t dir; - int i, e; - qboolean hitClient = qfalse; - - if( radius < 1 ) - radius = 1; - - for( i = 0; i < 3; i++ ) - { - mins[ i ] = origin[ i ] - radius; - maxs[ i ] = origin[ i ] + radius; - } - - numListedEntities = trap_EntitiesInBox( mins, maxs, entityList, MAX_GENTITIES ); - - for( e = 0; e < numListedEntities; e++ ) - { - ent = &g_entities[ entityList[ e ] ]; - - if( ent == ignore ) - continue; - - if( !ent->takedamage ) - continue; - - if( ent->flags & FL_NOTARGET ) - continue; - - // find the distance from the edge of the bounding box - for( i = 0 ; i < 3 ; i++ ) - { - if( origin[ i ] < ent->r.absmin[ i ] ) - v[ i ] = ent->r.absmin[ i ] - origin[ i ]; - else if( origin[ i ] > ent->r.absmax[ i ] ) - v[ i ] = origin[ i ] - ent->r.absmax[ i ]; - else - v[ i ] = 0; - } - - dist = VectorLength( v ); - if( dist >= radius ) - continue; - - points = damage * ( 1.0 - dist / radius ); - - if( CanDamage( ent, origin ) && ent->client && - ent->client->ps.stats[ STAT_TEAM ] != team ) - { - VectorSubtract( ent->r.currentOrigin, origin, dir ); - // push the center of mass higher than the origin so players - // get knocked into the air more - dir[ 2 ] += 24; - hitClient = qtrue; - G_Damage( ent, NULL, attacker, dir, origin, - (int)points, DAMAGE_RADIUS|DAMAGE_NO_LOCDAMAGE, mod ); - } - } - - return hitClient; -} - - -/* -============ -G_RadiusDamage -============ -*/ -qboolean G_RadiusDamage( vec3_t origin, gentity_t *attacker, float damage, - float radius, gentity_t *ignore, int mod ) -{ - float points, dist; - gentity_t *ent; - int entityList[ MAX_GENTITIES ]; - int numListedEntities; - vec3_t mins, maxs; - vec3_t v; - vec3_t dir; - int i, e; - qboolean hitClient = qfalse; - - if( radius < 1 ) - radius = 1; - - for( i = 0; i < 3; i++ ) - { - mins[ i ] = origin[ i ] - radius; - maxs[ i ] = origin[ i ] + radius; - } - - numListedEntities = trap_EntitiesInBox( mins, maxs, entityList, MAX_GENTITIES ); - - for( e = 0; e < numListedEntities; e++ ) - { - ent = &g_entities[ entityList[ e ] ]; - - if( ent == ignore ) - continue; - - if( !ent->takedamage ) - continue; - - // find the distance from the edge of the bounding box - for( i = 0; i < 3; i++ ) - { - if( origin[ i ] < ent->r.absmin[ i ] ) - v[ i ] = ent->r.absmin[ i ] - origin[ i ]; - else if( origin[ i ] > ent->r.absmax[ i ] ) - v[ i ] = origin[ i ] - ent->r.absmax[ i ]; - else - v[ i ] = 0; - } - - dist = VectorLength( v ); - if( dist >= radius ) - continue; - - points = damage * ( 1.0 - dist / radius ); - - if( CanDamage( ent, origin ) ) - { - VectorSubtract( ent->r.currentOrigin, origin, dir ); - // push the center of mass higher than the origin so players - // get knocked into the air more - dir[ 2 ] += 24; - hitClient = qtrue; - G_Damage( ent, NULL, attacker, dir, origin, - (int)points, DAMAGE_RADIUS|DAMAGE_NO_LOCDAMAGE, mod ); - } - } - - return hitClient; -} - -/* -================ -G_LogDestruction - -Log deconstruct/destroy events -================ -*/ -void G_LogDestruction( gentity_t *self, gentity_t *actor, int mod ) -{ - buildFate_t fate; - - switch( mod ) - { - case MOD_DECONSTRUCT: - fate = BF_DECONSTRUCT; - break; - case MOD_REPLACE: - fate = BF_REPLACE; - break; - case MOD_NOCREEP: - fate = ( actor->client ) ? BF_UNPOWER : BF_AUTO; - break; - default: - if( actor->client ) - { - if( actor->client->pers.teamSelection == - BG_Buildable( self->s.modelindex, NULL )->team ) - { - fate = BF_TEAMKILL; - } - else - fate = BF_DESTROY; - } - else - fate = BF_AUTO; - break; - } - G_BuildLogAuto( actor, self, fate ); - - // don't log when marked structures are removed - if( mod == MOD_REPLACE ) - return; - - G_LogPrintf( S_COLOR_YELLOW "Deconstruct: %d %d %s %s: %s %s by %s\n", - actor - g_entities, - self - g_entities, - BG_Buildable( self->s.modelindex, NULL )->name, - modNames[ mod ], - G_CuboidName(self->s.modelindex,self->cuboidSize,qtrue), - mod == MOD_DECONSTRUCT ? "deconstructed" : "destroyed", - actor->client ? actor->client->pers.netname : "<world>" ); - - // No-power deaths for humans come after some minutes and it's confusing - // when the messages appear attributed to the deconner. Just don't print them. - if( mod == MOD_NOCREEP && actor->client && - actor->client->pers.teamSelection == TEAM_HUMANS ) - return; - - if( actor->client && actor->client->pers.teamSelection == - BG_Buildable( self->s.modelindex, NULL )->team ) - { - G_TeamCommand( actor->client->ps.stats[ STAT_TEAM ], - va( "print \"%s ^3%s^7 by %s\n\"", - G_CuboidName(self->s.modelindex,self->cuboidSize,qfalse), - mod == MOD_DECONSTRUCT ? "DECONSTRUCTED" : "DESTROYED", - actor->client->pers.netname ) ); - } - -} diff --git a/src/game/g_main.c.orig b/src/game/g_main.c.orig deleted file mode 100644 index 33784c9..0000000 --- a/src/game/g_main.c.orig +++ /dev/null @@ -1,2548 +0,0 @@ -/* -=========================================================================== -Copyright (C) 1999-2005 Id Software, Inc. -Copyright (C) 2000-2009 Darklegion Development - -This file is part of Tremulous. - -Tremulous 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. - -Tremulous is distributed in the hope that it will be -useful, but WITHOUT ANY WARRANTY; without even the implied warranty of -MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -GNU General Public License for more details. - -You should have received a copy of the GNU General Public License -along with Tremulous; if not, write to the Free Software -Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA -=========================================================================== -*/ - -#include "g_local.h" - -level_locals_t level; - -typedef struct -{ - vmCvar_t *vmCvar; - char *cvarName; - char *defaultString; - int cvarFlags; - int modificationCount; // for tracking changes - qboolean trackChange; // track this variable, and announce if changed - /* certain cvars can be set in worldspawn, but we don't want those values to - persist, so keep track of non-worldspawn changes and restore that on map - end. unfortunately, if the server crashes, the value set in worldspawn may - persist */ - char *explicit; -} cvarTable_t; - -gentity_t g_entities[ MAX_GENTITIES ]; -gclient_t g_clients[ MAX_CLIENTS ]; - -vmCvar_t g_timelimit; -vmCvar_t g_suddenDeathTime; -vmCvar_t g_friendlyFire; -vmCvar_t g_friendlyBuildableFire; -vmCvar_t g_dretchPunt; -vmCvar_t g_password; -vmCvar_t g_needpass; -vmCvar_t g_maxclients; -vmCvar_t g_maxGameClients; -vmCvar_t g_dedicated; -vmCvar_t g_speed; -vmCvar_t g_gravity; -vmCvar_t g_cheats; -vmCvar_t g_knockback; -vmCvar_t g_inactivity; -vmCvar_t g_debugMove; -vmCvar_t g_debugDamage; -vmCvar_t g_motd; -vmCvar_t g_synchronousClients; -vmCvar_t g_warmup; -vmCvar_t g_doWarmup; -vmCvar_t g_restarted; -vmCvar_t g_lockTeamsAtStart; -vmCvar_t g_logFile; -vmCvar_t g_logFileSync; -vmCvar_t g_allowVote; -vmCvar_t g_voteLimit; -vmCvar_t g_suddenDeathVotePercent; -vmCvar_t g_suddenDeathVoteDelay; -vmCvar_t g_teamForceBalance; -vmCvar_t g_smoothClients; -vmCvar_t pmove_fixed; -vmCvar_t pmove_msec; -vmCvar_t g_minNameChangePeriod; -vmCvar_t g_maxNameChanges; - -vmCvar_t g_alienBuildPoints; -vmCvar_t g_alienBuildQueueTime; -vmCvar_t g_humanBuildPoints; -vmCvar_t g_humanBuildQueueTime; -vmCvar_t g_humanRepeaterBuildPoints; -vmCvar_t g_humanRepeaterBuildQueueTime; -vmCvar_t g_humanRepeaterMaxZones; -vmCvar_t g_humanStage; -vmCvar_t g_humanCredits; -vmCvar_t g_humanMaxStage; -vmCvar_t g_humanStage2Threshold; -vmCvar_t g_humanStage3Threshold; -vmCvar_t g_alienStage; -vmCvar_t g_alienCredits; -vmCvar_t g_alienMaxStage; -vmCvar_t g_alienStage2Threshold; -vmCvar_t g_alienStage3Threshold; -vmCvar_t g_teamImbalanceWarnings; -vmCvar_t g_freeFundPeriod; - -vmCvar_t g_unlagged; - -vmCvar_t g_disabledEquipment; -vmCvar_t g_disabledClasses; -vmCvar_t g_disabledBuildables; - -vmCvar_t g_markDeconstruct; - -vmCvar_t g_debugMapRotation; -vmCvar_t g_currentMapRotation; -vmCvar_t g_mapRotationNodes; -vmCvar_t g_mapRotationStack; -vmCvar_t g_nextMap; -vmCvar_t g_initialMapRotation; - -vmCvar_t g_debugVoices; -vmCvar_t g_voiceChats; - -vmCvar_t g_shove; - -vmCvar_t g_mapConfigs; -vmCvar_t g_sayAreaRange; - -vmCvar_t g_floodMaxDemerits; -vmCvar_t g_floodMinTime; - -vmCvar_t g_layouts; -vmCvar_t g_layoutAuto; - -vmCvar_t g_emoticonsAllowedInNames; - -vmCvar_t g_admin; -vmCvar_t g_adminTempBan; -vmCvar_t g_adminMaxBan; - -vmCvar_t g_privateMessages; -vmCvar_t g_specChat; -vmCvar_t g_publicAdminMessages; -vmCvar_t g_allowTeamOverlay; - -vmCvar_t g_censorship; - -vmCvar_t g_tag; - -vmCvar_t g_unlimited; -vmCvar_t g_instantBuild; -vmCvar_t g_cuboidSizeLimit; -vmCvar_t g_cuboidMode; - -vmCvar_t g_buildableDensityLimit; -vmCvar_t g_buildableDensityLimitRange; - -// copy cvars that can be set in worldspawn so they can be restored later -static char cv_gravity[ MAX_CVAR_VALUE_STRING ]; -static char cv_humanMaxStage[ MAX_CVAR_VALUE_STRING ]; -static char cv_alienMaxStage[ MAX_CVAR_VALUE_STRING ]; - -static cvarTable_t gameCvarTable[ ] = -{ - // don't override the cheat state set by the system - { &g_cheats, "sv_cheats", "", 0, 0, qfalse }, - - // noset vars - { NULL, "gamename", GAME_VERSION , CVAR_SERVERINFO | CVAR_ROM, 0, qfalse }, - { NULL, "gamedate", __DATE__ , CVAR_ROM, 0, qfalse }, - { &g_restarted, "g_restarted", "0", CVAR_ROM, 0, qfalse }, - { &g_lockTeamsAtStart, "g_lockTeamsAtStart", "0", CVAR_ROM, 0, qfalse }, - { NULL, "sv_mapname", "", CVAR_SERVERINFO | CVAR_ROM, 0, qfalse }, - { NULL, "P", "", CVAR_SERVERINFO | CVAR_ROM, 0, qfalse }, - - // latched vars - - { &g_maxclients, "sv_maxclients", "8", CVAR_SERVERINFO | CVAR_LATCH | CVAR_ARCHIVE, 0, qfalse }, - - // change anytime vars - { &g_maxGameClients, "g_maxGameClients", "0", CVAR_SERVERINFO | CVAR_ARCHIVE, 0, qfalse }, - - { &g_timelimit, "timelimit", "0", CVAR_SERVERINFO | CVAR_ARCHIVE | CVAR_NORESTART, 0, qtrue }, - { &g_suddenDeathTime, "g_suddenDeathTime", "0", CVAR_SERVERINFO | CVAR_ARCHIVE | CVAR_NORESTART, 0, qtrue }, - - { &g_synchronousClients, "g_synchronousClients", "0", CVAR_SYSTEMINFO, 0, qfalse }, - - { &g_friendlyFire, "g_friendlyFire", "0", CVAR_SERVERINFO | CVAR_ARCHIVE, 0, qtrue }, - { &g_friendlyBuildableFire, "g_friendlyBuildableFire", "0", CVAR_SERVERINFO | CVAR_ARCHIVE, 0, qtrue }, - { &g_dretchPunt, "g_dretchPunt", "1", CVAR_ARCHIVE, 0, qtrue }, - - { &g_teamForceBalance, "g_teamForceBalance", "0", CVAR_ARCHIVE, 0, qtrue }, - - { &g_warmup, "g_warmup", "10", CVAR_ARCHIVE, 0, qtrue }, - { &g_doWarmup, "g_doWarmup", "0", CVAR_ARCHIVE, 0, qtrue }, - { &g_logFile, "g_logFile", "games.log", CVAR_ARCHIVE, 0, qfalse }, - { &g_logFileSync, "g_logFileSync", "0", CVAR_ARCHIVE, 0, qfalse }, - - { &g_password, "g_password", "", CVAR_USERINFO, 0, qfalse }, - - { &g_needpass, "g_needpass", "0", CVAR_SERVERINFO | CVAR_ROM, 0, qfalse }, - - { &g_dedicated, "dedicated", "0", 0, 0, qfalse }, - - { &g_speed, "g_speed", "320", 0, 0, qtrue }, - { &g_gravity, "g_gravity", "800", 0, 0, qtrue, cv_gravity }, - { &g_knockback, "g_knockback", "1000", 0, 0, qtrue }, - { &g_inactivity, "g_inactivity", "0", 0, 0, qtrue }, - { &g_debugMove, "g_debugMove", "0", 0, 0, qfalse }, - { &g_debugDamage, "g_debugDamage", "0", 0, 0, qfalse }, - { &g_motd, "g_motd", "", 0, 0, qfalse }, - - { &g_allowVote, "g_allowVote", "1", CVAR_ARCHIVE, 0, qfalse }, - { &g_voteLimit, "g_voteLimit", "5", CVAR_ARCHIVE, 0, qfalse }, - { &g_suddenDeathVotePercent, "g_suddenDeathVotePercent", "74", CVAR_ARCHIVE, 0, qfalse }, - { &g_suddenDeathVoteDelay, "g_suddenDeathVoteDelay", "180", CVAR_ARCHIVE, 0, qfalse }, - { &g_minNameChangePeriod, "g_minNameChangePeriod", "5", 0, 0, qfalse}, - { &g_maxNameChanges, "g_maxNameChanges", "5", 0, 0, qfalse}, - - { &g_smoothClients, "g_smoothClients", "1", 0, 0, qfalse}, - { &pmove_fixed, "pmove_fixed", "0", CVAR_SYSTEMINFO, 0, qfalse}, - { &pmove_msec, "pmove_msec", "8", CVAR_SYSTEMINFO, 0, qfalse}, - - { &g_alienBuildPoints, "g_alienBuildPoints", DEFAULT_ALIEN_BUILDPOINTS, 0, 0, qfalse }, - { &g_alienBuildQueueTime, "g_alienBuildQueueTime", DEFAULT_ALIEN_QUEUE_TIME, CVAR_ARCHIVE, 0, qfalse }, - { &g_humanBuildPoints, "g_humanBuildPoints", DEFAULT_HUMAN_BUILDPOINTS, 0, 0, qfalse }, - { &g_humanBuildQueueTime, "g_humanBuildQueueTime", DEFAULT_HUMAN_QUEUE_TIME, CVAR_ARCHIVE, 0, qfalse }, - { &g_humanRepeaterBuildPoints, "g_humanRepeaterBuildPoints", DEFAULT_HUMAN_REPEATER_BUILDPOINTS, CVAR_ARCHIVE, 0, qfalse }, - { &g_humanRepeaterMaxZones, "g_humanRepeaterMaxZones", DEFAULT_HUMAN_REPEATER_MAX_ZONES, CVAR_ARCHIVE, 0, qfalse }, - { &g_humanRepeaterBuildQueueTime, "g_humanRepeaterBuildQueueTime", DEFAULT_HUMAN_REPEATER_QUEUE_TIME, CVAR_ARCHIVE, 0, qfalse }, - { &g_humanStage, "g_humanStage", "0", 0, 0, qfalse }, - { &g_humanCredits, "g_humanCredits", "0", 0, 0, qfalse }, - { &g_humanMaxStage, "g_humanMaxStage", DEFAULT_HUMAN_MAX_STAGE, 0, 0, qfalse, cv_humanMaxStage }, - { &g_humanStage2Threshold, "g_humanStage2Threshold", DEFAULT_HUMAN_STAGE2_THRESH, 0, 0, qfalse }, - { &g_humanStage3Threshold, "g_humanStage3Threshold", DEFAULT_HUMAN_STAGE3_THRESH, 0, 0, qfalse }, - { &g_alienStage, "g_alienStage", "0", 0, 0, qfalse }, - { &g_alienCredits, "g_alienCredits", "0", 0, 0, qfalse }, - { &g_alienMaxStage, "g_alienMaxStage", DEFAULT_ALIEN_MAX_STAGE, 0, 0, qfalse, cv_alienMaxStage }, - { &g_alienStage2Threshold, "g_alienStage2Threshold", DEFAULT_ALIEN_STAGE2_THRESH, 0, 0, qfalse }, - { &g_alienStage3Threshold, "g_alienStage3Threshold", DEFAULT_ALIEN_STAGE3_THRESH, 0, 0, qfalse }, - { &g_teamImbalanceWarnings, "g_teamImbalanceWarnings", "30", CVAR_ARCHIVE, 0, qfalse }, - { &g_freeFundPeriod, "g_freeFundPeriod", DEFAULT_FREEKILL_PERIOD, CVAR_ARCHIVE, 0, qtrue }, - - { &g_unlagged, "g_unlagged", "1", CVAR_SERVERINFO | CVAR_ARCHIVE, 0, qtrue }, - - { &g_disabledEquipment, "g_disabledEquipment", "", CVAR_ROM | CVAR_SYSTEMINFO, 0, qfalse }, - { &g_disabledClasses, "g_disabledClasses", "", CVAR_ROM | CVAR_SYSTEMINFO, 0, qfalse }, - { &g_disabledBuildables, "g_disabledBuildables", "", CVAR_ROM | CVAR_SYSTEMINFO, 0, qfalse }, - - { &g_sayAreaRange, "g_sayAreaRange", "1000", CVAR_ARCHIVE, 0, qtrue }, - - { &g_floodMaxDemerits, "g_floodMaxDemerits", "5000", CVAR_ARCHIVE, 0, qfalse }, - { &g_floodMinTime, "g_floodMinTime", "2000", CVAR_ARCHIVE, 0, qfalse }, - - { &g_markDeconstruct, "g_markDeconstruct", "3", CVAR_SERVERINFO | CVAR_ARCHIVE, 0, qtrue }, - - { &g_debugMapRotation, "g_debugMapRotation", "0", 0, 0, qfalse }, - { &g_currentMapRotation, "g_currentMapRotation", "-1", 0, 0, qfalse }, // -1 = NOT_ROTATING - { &g_mapRotationNodes, "g_mapRotationNodes", "", CVAR_ROM, 0, qfalse }, - { &g_mapRotationStack, "g_mapRotationStack", "", CVAR_ROM, 0, qfalse }, - { &g_nextMap, "g_nextMap", "", 0 , 0, qtrue }, - { &g_initialMapRotation, "g_initialMapRotation", "", CVAR_ARCHIVE, 0, qfalse }, - { &g_debugVoices, "g_debugVoices", "0", 0, 0, qfalse }, - { &g_voiceChats, "g_voiceChats", "1", CVAR_ARCHIVE, 0, qfalse }, - { &g_shove, "g_shove", "0.0", CVAR_ARCHIVE, 0, qfalse }, - { &g_mapConfigs, "g_mapConfigs", "", CVAR_ARCHIVE, 0, qfalse }, - { NULL, "g_mapConfigsLoaded", "0", CVAR_ROM, 0, qfalse }, - - { &g_layouts, "g_layouts", "", CVAR_LATCH, 0, qfalse }, - { &g_layoutAuto, "g_layoutAuto", "1", CVAR_ARCHIVE, 0, qfalse }, - - { &g_emoticonsAllowedInNames, "g_emoticonsAllowedInNames", "1", CVAR_LATCH|CVAR_ARCHIVE, 0, qfalse }, - - { &g_admin, "g_admin", "admin.dat", CVAR_ARCHIVE, 0, qfalse }, - { &g_adminTempBan, "g_adminTempBan", "2m", CVAR_ARCHIVE, 0, qfalse }, - { &g_adminMaxBan, "g_adminMaxBan", "2w", CVAR_ARCHIVE, 0, qfalse }, - - { &g_privateMessages, "g_privateMessages", "1", CVAR_ARCHIVE, 0, qfalse }, - { &g_specChat, "g_specChat", "1", CVAR_ARCHIVE, 0, qfalse }, - { &g_publicAdminMessages, "g_publicAdminMessages", "1", CVAR_ARCHIVE, 0, qfalse }, - { &g_allowTeamOverlay, "g_allowTeamOverlay", "1", CVAR_ARCHIVE, 0, qtrue }, - - { &g_censorship, "g_censorship", "", CVAR_ARCHIVE, 0, qfalse }, - - { &g_tag, "g_tag", "gpp", CVAR_INIT, 0, qfalse }, - - { &g_unlimited, "g_unlimited", "0", CVAR_ARCHIVE | CVAR_SERVERINFO, 0, qfalse }, - { &g_instantBuild, "g_instantBuild", "0", CVAR_ARCHIVE | CVAR_SERVERINFO, 0, qfalse }, - { &g_cuboidSizeLimit, "g_cuboidSizeLimit", "0", CVAR_ARCHIVE, 0, qfalse }, - { &g_cuboidMode, "g_cuboidMode", "0", CVAR_ARCHIVE, 0, qfalse }, - - { &g_buildableDensityLimit, "g_buildableDensityLimit", "0", CVAR_ARCHIVE, 0, qfalse }, - { &g_buildableDensityLimitRange, "g_buildableDensityLimitRange", "0", CVAR_ARCHIVE, 0, qfalse } -}; - -static int gameCvarTableSize = sizeof( gameCvarTable ) / sizeof( gameCvarTable[ 0 ] ); - - -void G_InitGame( int levelTime, int randomSeed, int restart ); -void G_RunFrame( int levelTime ); -void G_ShutdownGame( int restart ); -void CheckExitRules( void ); - -void G_CountSpawns( void ); -void G_CalculateBuildPoints( void ); - -/* -================ -vmMain - -This is the only way control passes into the module. -This must be the very first function compiled into the .q3vm file -================ -*/ -Q_EXPORT intptr_t vmMain( int command, int arg0, int arg1, int arg2, int arg3, int arg4, - int arg5, int arg6, int arg7, int arg8, int arg9, - int arg10, int arg11 ) -{ - switch( command ) - { - case GAME_INIT: - G_InitGame( arg0, arg1, arg2 ); - return 0; - - case GAME_SHUTDOWN: - G_ShutdownGame( arg0 ); - return 0; - - case GAME_CLIENT_CONNECT: - return (intptr_t)ClientConnect( arg0, arg1 ); - - case GAME_CLIENT_THINK: - ClientThink( arg0 ); - return 0; - - case GAME_CLIENT_USERINFO_CHANGED: - ClientUserinfoChanged( arg0, qfalse ); - return 0; - - case GAME_CLIENT_DISCONNECT: - ClientDisconnect( arg0 ); - return 0; - - case GAME_CLIENT_BEGIN: - ClientBegin( arg0 ); - return 0; - - case GAME_CLIENT_COMMAND: - ClientCommand( arg0 ); - return 0; - - case GAME_RUN_FRAME: - G_RunFrame( arg0 ); - return 0; - - case GAME_CONSOLE_COMMAND: - return ConsoleCommand( ); - } - - return -1; -} - - -void QDECL G_Printf( const char *fmt, ... ) -{ - va_list argptr; - char text[ 1024 ]; - - va_start( argptr, fmt ); - Q_vsnprintf( text, sizeof( text ), fmt, argptr ); - va_end( argptr ); - - trap_Print( text ); -} - -void QDECL G_Error( const char *fmt, ... ) -{ - va_list argptr; - char text[ 1024 ]; - - va_start( argptr, fmt ); - Q_vsnprintf( text, sizeof( text ), fmt, argptr ); - va_end( argptr ); - - trap_Error( text ); -} - -/* -================ -G_FindTeams - -Chain together all entities with a matching team field. -Entity teams are used for item groups and multi-entity mover groups. - -All but the first will have the FL_TEAMSLAVE flag set and teammaster field set -All but the last will have the teamchain field set to the next one -================ -*/ -void G_FindTeams( void ) -{ - gentity_t *e, *e2; - int i, j; - int c, c2; - - c = 0; - c2 = 0; - - for( i = 1, e = g_entities+i; i < level.num_entities; i++, e++ ) - { - if( !e->inuse ) - continue; - - if( !e->team ) - continue; - - if( e->flags & FL_TEAMSLAVE ) - continue; - - e->teammaster = e; - c++; - c2++; - - for( j = i + 1, e2 = e + 1; j < level.num_entities; j++, e2++ ) - { - if( !e2->inuse ) - continue; - - if( !e2->team ) - continue; - - if( e2->flags & FL_TEAMSLAVE ) - continue; - - if( !strcmp( e->team, e2->team ) ) - { - c2++; - e2->teamchain = e->teamchain; - e->teamchain = e2; - e2->teammaster = e; - e2->flags |= FL_TEAMSLAVE; - - // make sure that targets only point at the master - if( e2->targetname ) - { - e->targetname = e2->targetname; - e2->targetname = NULL; - } - } - } - } - - G_Printf( "%i teams with %i entities\n", c, c2 ); -} - - -/* -================= -G_RegisterCvars -================= -*/ -void G_RegisterCvars( void ) -{ - int i; - cvarTable_t *cv; - - for( i = 0, cv = gameCvarTable; i < gameCvarTableSize; i++, cv++ ) - { - trap_Cvar_Register( cv->vmCvar, cv->cvarName, - cv->defaultString, cv->cvarFlags ); - - if( cv->vmCvar ) - cv->modificationCount = cv->vmCvar->modificationCount; - - if( cv->explicit ) - strcpy( cv->explicit, cv->vmCvar->string ); - } -} - -/* -================= -G_UpdateCvars -================= -*/ -void G_UpdateCvars( void ) -{ - int i; - cvarTable_t *cv; - - for( i = 0, cv = gameCvarTable; i < gameCvarTableSize; i++, cv++ ) - { - if( cv->vmCvar ) - { - trap_Cvar_Update( cv->vmCvar ); - - if( cv->modificationCount != cv->vmCvar->modificationCount ) - { - cv->modificationCount = cv->vmCvar->modificationCount; - - if( cv->trackChange ) - trap_SendServerCommand( -1, va( "print \"Server: %s changed to %s\n\"", - cv->cvarName, cv->vmCvar->string ) ); - - if( !level.spawning && cv->explicit ) - strcpy( cv->explicit, cv->vmCvar->string ); - } - } - } -} - -/* -================= -G_RestoreCvars -================= -*/ -void G_RestoreCvars( void ) -{ - int i; - cvarTable_t *cv; - - for( i = 0, cv = gameCvarTable; i < gameCvarTableSize; i++, cv++ ) - { - if( cv->vmCvar && cv->explicit ) - trap_Cvar_Set( cv->cvarName, cv->explicit ); - } -} - -/* -================= -G_MapConfigs -================= -*/ -void G_MapConfigs( const char *mapname ) -{ - - if( !g_mapConfigs.string[0] ) - return; - - if( trap_Cvar_VariableIntegerValue( "g_mapConfigsLoaded" ) ) - return; - - trap_SendConsoleCommand( EXEC_APPEND, - va( "exec \"%s/default.cfg\"\n", g_mapConfigs.string ) ); - - trap_SendConsoleCommand( EXEC_APPEND, - va( "exec \"%s/%s.cfg\"\n", g_mapConfigs.string, mapname ) ); - - trap_Cvar_Set( "g_mapConfigsLoaded", "1" ); -} - -/* -================ -G_CheckCuboidConfig - -even tokens are map names -odd tokens are cuboid modes -================ -*/ -void G_CheckCuboidConfig(char* mapname) -{ - char* token; - char config[MAX_CVAR_VALUE_STRING]; //should be enough for few maps - qboolean type=qfalse; - qboolean found=qfalse; - int mode; - - mode=trap_Cvar_VariableIntegerValue("g_cuboidDefaultMode"); - trap_Cvar_VariableStringBuffer("g_cuboidConfig",config,sizeof(config)); - - while(1) - { - COM_Parse(&token); - - if(!token) - break; - - if(!type) - if(!Q_stricmp(token,mapname)) - found=qtrue; - else - if(found) - { - mode=atoi(token); - break; - } - } - Com_Printf("cuboids are %s%s^7 on %s\n",(mode==0?"^2ENABLED":(mode==1?"^1DISABLED ON S1":"^1DISABLED")),(found?"":" BY DEFAULT"),mapname); -} - -/* -============ -G_InitGame - -============ -*/ -void G_InitGame( int levelTime, int randomSeed, int restart ) -{ - int i; - char map[ MAX_CVAR_VALUE_STRING ] = {""}; - - srand( randomSeed ); - - G_RegisterCvars( ); - - G_Printf( "------- Game Initialization -------\n" ); - G_Printf( "gamename: %s\n", GAME_VERSION ); - G_Printf( "gamedate: %s\n", __DATE__ ); - - BG_InitMemory( ); - - // set some level globals - memset( &level, 0, sizeof( level ) ); - level.time = levelTime; - level.startTime = levelTime; - level.alienStage2Time = level.alienStage3Time = - level.humanStage2Time = level.humanStage3Time = level.startTime; - - level.snd_fry = G_SoundIndex( "sound/misc/fry.wav" ); // FIXME standing in lava / slime - - if( g_logFile.string[ 0 ] ) - { - if( g_logFileSync.integer ) - trap_FS_FOpenFile( g_logFile.string, &level.logFile, FS_APPEND_SYNC ); - else - trap_FS_FOpenFile( g_logFile.string, &level.logFile, FS_APPEND ); - - if( !level.logFile ) - G_Printf( "WARNING: Couldn't open logfile: %s\n", g_logFile.string ); - else - { - char serverinfo[ MAX_INFO_STRING ]; - qtime_t qt; - int t; - - trap_GetServerinfo( serverinfo, sizeof( serverinfo ) ); - - G_LogPrintf( "------------------------------------------------------------\n" ); - G_LogPrintf( "InitGame: %s\n", serverinfo ); - - t = trap_RealTime( &qt ); - G_LogPrintf("RealTime: %04i/%02i/%02i %02i:%02i:%02i\n", - qt.tm_year+1900, qt.tm_mon+1, qt.tm_mday, - qt.tm_hour, qt.tm_min, qt.tm_sec ); - - } - } - else - G_Printf( "Not logging to disk\n" ); - - trap_Cvar_VariableStringBuffer( "mapname", map, sizeof( map ) ); - G_MapConfigs( map ); - - // we're done with g_mapConfigs, so reset this for the next map - trap_Cvar_Set( "g_mapConfigsLoaded", "0" ); - - G_RegisterCommands( ); - G_admin_readconfig( NULL ); - G_LoadCensors( ); - - // initialize all entities for this game - memset( g_entities, 0, MAX_GENTITIES * sizeof( g_entities[ 0 ] ) ); - level.gentities = g_entities; - - // initialize all clients for this game - level.maxclients = g_maxclients.integer; - memset( g_clients, 0, MAX_CLIENTS * sizeof( g_clients[ 0 ] ) ); - level.clients = g_clients; - - // set client fields on player ents - for( i = 0; i < level.maxclients; i++ ) - g_entities[ i ].client = level.clients + i; - - // always leave room for the max number of clients, - // even if they aren't all used, so numbers inside that - // range are NEVER anything but clients - level.num_entities = MAX_CLIENTS; - - // let the server system know where the entites are - trap_LocateGameData( level.gentities, level.num_entities, sizeof( gentity_t ), - &level.clients[ 0 ].ps, sizeof( level.clients[ 0 ] ) ); - - level.emoticonCount = BG_LoadEmoticons( level.emoticons, MAX_EMOTICONS ); - - trap_SetConfigstring( CS_INTERMISSION, "0" ); - - // test to see if a custom buildable layout will be loaded - G_LayoutSelect( ); - - // this has to be flipped after the first UpdateCvars - level.spawning = qtrue; - // parse the key/value pairs and spawn gentities - G_SpawnEntitiesFromString( ); - - // load up a custom building layout if there is one - G_LayoutLoad( ); - - // find out g_cuboidMode value - G_CheckCuboidConfig( map ); - - // the map might disable some things - BG_InitAllowedGameElements( ); - - // general initialization - G_FindTeams( ); - - BG_InitClassConfigs( ); - BG_InitBuildableConfigs( ); - G_InitDamageLocations( ); - G_InitMapRotations( ); - G_InitSpawnQueue( &level.alienSpawnQueue ); - G_InitSpawnQueue( &level.humanSpawnQueue ); - - if( g_debugMapRotation.integer ) - G_PrintRotations( ); - - level.voices = BG_VoiceInit( ); - BG_PrintVoices( level.voices, g_debugVoices.integer ); - - //reset stages - trap_Cvar_Set( "g_alienStage", va( "%d", S1 ) ); - trap_Cvar_Set( "g_humanStage", va( "%d", S1 ) ); - trap_Cvar_Set( "g_alienCredits", 0 ); - trap_Cvar_Set( "g_humanCredits", 0 ); - level.suddenDeathBeginTime = g_suddenDeathTime.integer * 60000; - - G_Printf( "-----------------------------------\n" ); - - // So the server counts the spawns without a client attached - G_CountSpawns( ); - - G_UpdateTeamConfigStrings( ); - - if( g_lockTeamsAtStart.integer ) - { - level.alienTeamLocked = qtrue; - level.humanTeamLocked = qtrue; - trap_Cvar_Set( "g_lockTeamsAtStart", "0" ); - } -} - -/* -================== -G_ClearVotes - -remove all currently active votes -================== -*/ -static void G_ClearVotes( void ) -{ - int i; - memset( level.voteTime, 0, sizeof( level.voteTime ) ); - for( i = 0; i < NUM_TEAMS; i++ ) - { - trap_SetConfigstring( CS_VOTE_TIME + i, "" ); - trap_SetConfigstring( CS_VOTE_STRING + i, "" ); - } -} - -/* -================= -G_ShutdownGame -================= -*/ -void G_ShutdownGame( int restart ) -{ - // in case of a map_restart - G_ClearVotes( ); - - G_RestoreCvars( ); - - G_Printf( "==== ShutdownGame ====\n" ); - - if( level.logFile ) - { - G_LogPrintf( "ShutdownGame:\n" ); - G_LogPrintf( "------------------------------------------------------------\n" ); - trap_FS_FCloseFile( level.logFile ); - level.logFile = 0; - } - - // write all the client session data so we can get it back - G_WriteSessionData( ); - - G_admin_cleanup( ); - G_namelog_cleanup( ); - G_UnregisterCommands( ); - - G_ShutdownMapRotations( ); - - level.restarted = qfalse; - level.surrenderTeam = TEAM_NONE; - trap_SetConfigstring( CS_WINNER, "" ); -} - - - -//=================================================================== - -void QDECL Com_Error( int level, const char *error, ... ) -{ - va_list argptr; - char text[ 1024 ]; - - va_start( argptr, error ); - Q_vsnprintf( text, sizeof( text ), error, argptr ); - va_end( argptr ); - - G_Error( "%s", text ); -} - -void QDECL Com_Printf( const char *msg, ... ) -{ - va_list argptr; - char text[ 1024 ]; - - va_start( argptr, msg ); - Q_vsnprintf( text, sizeof( text ), msg, argptr ); - va_end( argptr ); - - G_Printf( "%s", text ); -} - -/* -======================================================================== - -PLAYER COUNTING / SCORE SORTING - -======================================================================== -*/ - - -/* -============= -SortRanks - -============= -*/ -int QDECL SortRanks( const void *a, const void *b ) -{ - gclient_t *ca, *cb; - - ca = &level.clients[ *(int *)a ]; - cb = &level.clients[ *(int *)b ]; - - // then sort by score - if( ca->ps.persistant[ PERS_SCORE ] > cb->ps.persistant[ PERS_SCORE ] ) - return -1; - if( ca->ps.persistant[ PERS_SCORE ] < cb->ps.persistant[ PERS_SCORE ] ) - return 1; - else - return 0; -} - -/* -============ -G_InitSpawnQueue - -Initialise a spawn queue -============ -*/ -void G_InitSpawnQueue( spawnQueue_t *sq ) -{ - int i; - - sq->back = sq->front = 0; - sq->back = QUEUE_MINUS1( sq->back ); - - //0 is a valid clientNum, so use something else - for( i = 0; i < MAX_CLIENTS; i++ ) - sq->clients[ i ] = -1; -} - -/* -============ -G_GetSpawnQueueLength - -Return the length of a spawn queue -============ -*/ -int G_GetSpawnQueueLength( spawnQueue_t *sq ) -{ - int length = sq->back - sq->front + 1; - - while( length < 0 ) - length += MAX_CLIENTS; - - while( length >= MAX_CLIENTS ) - length -= MAX_CLIENTS; - - return length; -} - -/* -============ -G_PopSpawnQueue - -Remove from front element from a spawn queue -============ -*/ -int G_PopSpawnQueue( spawnQueue_t *sq ) -{ - int clientNum = sq->clients[ sq->front ]; - - if( G_GetSpawnQueueLength( sq ) > 0 ) - { - sq->clients[ sq->front ] = -1; - sq->front = QUEUE_PLUS1( sq->front ); - G_StopFollowing( g_entities + clientNum ); - g_entities[ clientNum ].client->ps.pm_flags &= ~PMF_QUEUED; - - return clientNum; - } - else - return -1; -} - -/* -============ -G_PeekSpawnQueue - -Look at front element from a spawn queue -============ -*/ -int G_PeekSpawnQueue( spawnQueue_t *sq ) -{ - return sq->clients[ sq->front ]; -} - -/* -============ -G_SearchSpawnQueue - -Look to see if clientNum is already in the spawnQueue -============ -*/ -qboolean G_SearchSpawnQueue( spawnQueue_t *sq, int clientNum ) -{ - int i; - - for( i = 0; i < MAX_CLIENTS; i++ ) - { - if( sq->clients[ i ] == clientNum ) - return qtrue; - } - - return qfalse; -} - -/* -============ -G_PushSpawnQueue - -Add an element to the back of the spawn queue -============ -*/ -qboolean G_PushSpawnQueue( spawnQueue_t *sq, int clientNum ) -{ - // don't add the same client more than once - if( G_SearchSpawnQueue( sq, clientNum ) ) - return qfalse; - - sq->back = QUEUE_PLUS1( sq->back ); - sq->clients[ sq->back ] = clientNum; - - g_entities[ clientNum ].client->ps.pm_flags |= PMF_QUEUED; - return qtrue; -} - -/* -============ -G_RemoveFromSpawnQueue - -remove a specific client from a spawn queue -============ -*/ -qboolean G_RemoveFromSpawnQueue( spawnQueue_t *sq, int clientNum ) -{ - int i = sq->front; - - if( G_GetSpawnQueueLength( sq ) ) - { - do - { - if( sq->clients[ i ] == clientNum ) - { - //and this kids is why it would have - //been better to use an LL for internal - //representation - do - { - sq->clients[ i ] = sq->clients[ QUEUE_PLUS1( i ) ]; - - i = QUEUE_PLUS1( i ); - } while( i != QUEUE_PLUS1( sq->back ) ); - - sq->back = QUEUE_MINUS1( sq->back ); - g_entities[ clientNum ].client->ps.pm_flags &= ~PMF_QUEUED; - - return qtrue; - } - - i = QUEUE_PLUS1( i ); - } while( i != QUEUE_PLUS1( sq->back ) ); - } - - return qfalse; -} - -/* -============ -G_GetPosInSpawnQueue - -Get the position of a client in a spawn queue -============ -*/ -int G_GetPosInSpawnQueue( spawnQueue_t *sq, int clientNum ) -{ - int i = sq->front; - - if( G_GetSpawnQueueLength( sq ) ) - { - do - { - if( sq->clients[ i ] == clientNum ) - { - if( i < sq->front ) - return i + MAX_CLIENTS - sq->front; - else - return i - sq->front; - } - - i = QUEUE_PLUS1( i ); - } while( i != QUEUE_PLUS1( sq->back ) ); - } - - return -1; -} - -/* -============ -G_PrintSpawnQueue - -Print the contents of a spawn queue -============ -*/ -void G_PrintSpawnQueue( spawnQueue_t *sq ) -{ - int i = sq->front; - int length = G_GetSpawnQueueLength( sq ); - - G_Printf( "l:%d f:%d b:%d :", length, sq->front, sq->back ); - - if( length > 0 ) - { - do - { - if( sq->clients[ i ] == -1 ) - G_Printf( "*:" ); - else - G_Printf( "%d:", sq->clients[ i ] ); - - i = QUEUE_PLUS1( i ); - } while( i != QUEUE_PLUS1( sq->back ) ); - } - - G_Printf( "\n" ); -} - -/* -============ -G_SpawnClients - -Spawn queued clients -============ -*/ -void G_SpawnClients( team_t team ) -{ - int clientNum; - gentity_t *ent, *spawn; - vec3_t spawn_origin, spawn_angles; - spawnQueue_t *sq = NULL; - int numSpawns = 0; - - if( team == TEAM_ALIENS ) - { - sq = &level.alienSpawnQueue; - numSpawns = level.numAlienSpawns; - } - else if( team == TEAM_HUMANS ) - { - sq = &level.humanSpawnQueue; - numSpawns = level.numHumanSpawns; - } - - if( G_GetSpawnQueueLength( sq ) > 0 && numSpawns > 0 ) - { - clientNum = G_PeekSpawnQueue( sq ); - ent = &g_entities[ clientNum ]; - - if( ( spawn = G_SelectTremulousSpawnPoint( team, - ent->client->pers.lastDeathLocation, - spawn_origin, spawn_angles ) ) ) - { - clientNum = G_PopSpawnQueue( sq ); - - if( clientNum < 0 ) - return; - - ent = &g_entities[ clientNum ]; - - ent->client->sess.spectatorState = SPECTATOR_NOT; - ClientUserinfoChanged( clientNum, qfalse ); - ClientSpawn( ent, spawn, spawn_origin, spawn_angles ); - } - } -} - -/* -============ -G_CountSpawns - -Counts the number of spawns for each team -============ -*/ -void G_CountSpawns( void ) -{ - int i; - gentity_t *ent; - - level.numAlienSpawns = 0; - level.numHumanSpawns = 0; - for( i = 1, ent = g_entities + i ; i < level.num_entities ; i++, ent++ ) - { - if( !ent->inuse || ent->s.eType != ET_BUILDABLE || ent->health <= 0 ) - continue; - - if( ent->s.modelindex == BA_A_SPAWN ) - level.numAlienSpawns++; - - if( ent->s.modelindex == BA_H_SPAWN ) - level.numHumanSpawns++; - } -} - - -/* -============ -G_TimeTilSuddenDeath -============ -*/ -#define SUDDENDEATHWARNING 60000 -int G_TimeTilSuddenDeath( void ) -{ - if( ( !g_suddenDeathTime.integer && level.suddenDeathBeginTime == 0 ) || - ( level.suddenDeathBeginTime < 0 ) ) - return SUDDENDEATHWARNING + 1; // Always some time away - - return ( ( level.suddenDeathBeginTime ) - ( level.time - level.startTime ) ); -} - - -#define PLAYER_COUNT_MOD 5.0f - -/* -============ -G_CalculateBuildPoints - -Recalculate the quantity of building points available to the teams -============ -*/ -void G_CalculateBuildPoints( void ) -{ - int i; - buildable_t buildable; - buildPointZone_t *zone; - - // BP queue updates - while( level.alienBuildPointQueue > 0 && - level.alienNextQueueTime < level.time ) - { - level.alienBuildPointQueue--; - level.alienNextQueueTime += G_NextQueueTime( level.alienBuildPointQueue, - g_alienBuildPoints.integer, - g_alienBuildQueueTime.integer ); - } - - while( level.humanBuildPointQueue > 0 && - level.humanNextQueueTime < level.time ) - { - level.humanBuildPointQueue--; - level.humanNextQueueTime += G_NextQueueTime( level.humanBuildPointQueue, - g_humanBuildPoints.integer, - g_humanBuildQueueTime.integer ); - } - - // Sudden Death checks - if( G_TimeTilSuddenDeath( ) <= 0 && level.suddenDeathWarning < TW_PASSED ) - { - G_LogPrintf( "Beginning Sudden Death\n" ); - trap_SendServerCommand( -1, "cp \"Sudden Death!\"" ); - trap_SendServerCommand( -1, "print \"Beginning Sudden Death.\n\"" ); - level.suddenDeathWarning = TW_PASSED; - G_ClearDeconMarks( ); - - // Clear blueprints, or else structs that cost 0 BP can still be built after SD - for( i = 0; i < level.maxclients; i++ ) - { - if( g_entities[ i ].client->ps.stats[ STAT_BUILDABLE ] != BA_NONE ) - g_entities[ i ].client->ps.stats[ STAT_BUILDABLE ] = BA_NONE; - } - } - else if( G_TimeTilSuddenDeath( ) <= SUDDENDEATHWARNING && - level.suddenDeathWarning < TW_IMMINENT ) - { - trap_SendServerCommand( -1, va( "cp \"Sudden Death in %d seconds!\"", - (int)( G_TimeTilSuddenDeath( ) / 1000 ) ) ); - trap_SendServerCommand( -1, va( "print \"Sudden Death will begin in %d seconds.\n\"", - (int)( G_TimeTilSuddenDeath( ) / 1000 ) ) ); - level.suddenDeathWarning = TW_IMMINENT; - } - - level.humanBuildPoints = g_humanBuildPoints.integer - level.humanBuildPointQueue; - level.alienBuildPoints = g_alienBuildPoints.integer - level.alienBuildPointQueue; - - // Reset buildPointZones - for( i = 0; i < g_humanRepeaterMaxZones.integer; i++ ) - { - buildPointZone_t *zone = &level.buildPointZones[ i ]; - - zone->active = qfalse; - zone->totalBuildPoints = g_humanRepeaterBuildPoints.integer; - } - - // Iterate through entities - for( i = MAX_CLIENTS; i < level.num_entities; i++ ) - { - gentity_t *ent = &g_entities[ i ]; - buildPointZone_t *zone; - buildable_t buildable; - int cost; - - if( ent->s.eType != ET_BUILDABLE || ent->s.eFlags & EF_DEAD ) - continue; - - // mark a zone as active - if( ent->usesBuildPointZone ) - { - assert( ent->buildPointZone >= 0 && ent->buildPointZone < g_humanRepeaterMaxZones.integer ); - - zone = &level.buildPointZones[ ent->buildPointZone ]; - zone->active = qtrue; - } - - // Subtract the BP from the appropriate pool - buildable = ent->s.modelindex; - cost = BG_Buildable( buildable, ent->cuboidSize )->buildPoints; - - if( ent->buildableTeam == TEAM_ALIENS ) - level.alienBuildPoints -= cost; - if( buildable == BA_H_REPEATER ) - level.humanBuildPoints -= cost; - else if( buildable != BA_H_REACTOR ) - { - gentity_t *power = G_PowerEntityForEntity( ent ); - - if( power ) - { - if( power->s.modelindex == BA_H_REACTOR ) - level.humanBuildPoints -= cost; - else if( power->s.modelindex == BA_H_REPEATER && power->usesBuildPointZone ) - level.buildPointZones[ power->buildPointZone ].totalBuildPoints -= cost; - } - } - } - - // Finally, update repeater zones and their queues - // note that this has to be done after the used BP is calculated - for( i = MAX_CLIENTS; i < level.num_entities; i++ ) - { - gentity_t *ent = &g_entities[ i ]; - - if( ent->s.eType != ET_BUILDABLE || ent->s.eFlags & EF_DEAD || - ent->buildableTeam != TEAM_HUMANS ) - continue; - - buildable = ent->s.modelindex; - - if( buildable != BA_H_REPEATER ) - continue; - - if( ent->usesBuildPointZone && level.buildPointZones[ ent->buildPointZone ].active ) - { - zone = &level.buildPointZones[ ent->buildPointZone ]; - - if( G_TimeTilSuddenDeath( ) > 0 ) - { - // BP queue updates - while( zone->queuedBuildPoints > 0 && - zone->nextQueueTime < level.time ) - { - zone->queuedBuildPoints--; - zone->nextQueueTime += G_NextQueueTime( zone->queuedBuildPoints, - zone->totalBuildPoints, - g_humanRepeaterBuildQueueTime.integer ); - } - } - else - { - zone->totalBuildPoints = zone->queuedBuildPoints = 0; - } - } - } - - if( level.humanBuildPoints < 0 ) - level.humanBuildPoints = 0; - - if( level.alienBuildPoints < 0 ) - level.alienBuildPoints = 0; -} - -/* -============ -G_CalculateStages -============ -*/ -void G_CalculateStages( void ) -{ - float alienPlayerCountMod = level.averageNumAlienClients / PLAYER_COUNT_MOD; - float humanPlayerCountMod = level.averageNumHumanClients / PLAYER_COUNT_MOD; - int alienNextStageThreshold, humanNextStageThreshold; - static int lastAlienStageModCount = 1; - static int lastHumanStageModCount = 1; - - if( alienPlayerCountMod < 0.1f ) - alienPlayerCountMod = 0.1f; - - if( humanPlayerCountMod < 0.1f ) - humanPlayerCountMod = 0.1f; - - if( g_alienCredits.integer >= - (int)( ceil( (float)g_alienStage2Threshold.integer * alienPlayerCountMod ) ) && - g_alienStage.integer == S1 && g_alienMaxStage.integer > S1 ) - { - trap_Cvar_Set( "g_alienStage", va( "%d", S2 ) ); - level.alienStage2Time = level.time; - lastAlienStageModCount = g_alienStage.modificationCount; - G_LogPrintf("Stage: A 2: Aliens reached Stage 2\n"); - } - - if( g_alienCredits.integer >= - (int)( ceil( (float)g_alienStage3Threshold.integer * alienPlayerCountMod ) ) && - g_alienStage.integer == S2 && g_alienMaxStage.integer > S2 ) - { - trap_Cvar_Set( "g_alienStage", va( "%d", S3 ) ); - level.alienStage3Time = level.time; - lastAlienStageModCount = g_alienStage.modificationCount; - G_LogPrintf("Stage: A 3: Aliens reached Stage 3\n"); - } - - if( g_humanCredits.integer >= - (int)( ceil( (float)g_humanStage2Threshold.integer * humanPlayerCountMod ) ) && - g_humanStage.integer == S1 && g_humanMaxStage.integer > S1 ) - { - trap_Cvar_Set( "g_humanStage", va( "%d", S2 ) ); - level.humanStage2Time = level.time; - lastHumanStageModCount = g_humanStage.modificationCount; - G_LogPrintf("Stage: H 2: Humans reached Stage 2\n"); - } - - if( g_humanCredits.integer >= - (int)( ceil( (float)g_humanStage3Threshold.integer * humanPlayerCountMod ) ) && - g_humanStage.integer == S2 && g_humanMaxStage.integer > S2 ) - { - trap_Cvar_Set( "g_humanStage", va( "%d", S3 ) ); - level.humanStage3Time = level.time; - lastHumanStageModCount = g_humanStage.modificationCount; - G_LogPrintf("Stage: H 3: Humans reached Stage 3\n"); - } - - if( g_alienStage.modificationCount > lastAlienStageModCount ) - { - G_Checktrigger_stages( TEAM_ALIENS, g_alienStage.integer ); - - if( g_alienStage.integer == S2 ) - level.alienStage2Time = level.time; - else if( g_alienStage.integer == S3 ) - level.alienStage3Time = level.time; - - lastAlienStageModCount = g_alienStage.modificationCount; - } - - if( g_humanStage.modificationCount > lastHumanStageModCount ) - { - G_Checktrigger_stages( TEAM_HUMANS, g_humanStage.integer ); - - if( g_humanStage.integer == S2 ) - level.humanStage2Time = level.time; - else if( g_humanStage.integer == S3 ) - level.humanStage3Time = level.time; - - lastHumanStageModCount = g_humanStage.modificationCount; - } - - if( g_alienStage.integer == S1 && g_alienMaxStage.integer > S1 ) - alienNextStageThreshold = (int)( ceil( (float)g_alienStage2Threshold.integer * alienPlayerCountMod ) ); - else if( g_alienStage.integer == S2 && g_alienMaxStage.integer > S2 ) - alienNextStageThreshold = (int)( ceil( (float)g_alienStage3Threshold.integer * alienPlayerCountMod ) ); - else - alienNextStageThreshold = -1; - - if( g_humanStage.integer == S1 && g_humanMaxStage.integer > S1 ) - humanNextStageThreshold = (int)( ceil( (float)g_humanStage2Threshold.integer * humanPlayerCountMod ) ); - else if( g_humanStage.integer == S2 && g_humanMaxStage.integer > S2 ) - humanNextStageThreshold = (int)( ceil( (float)g_humanStage3Threshold.integer * humanPlayerCountMod ) ); - else - humanNextStageThreshold = -1; - - // save a lot of bandwidth by rounding thresholds up to the nearest 100 - if( alienNextStageThreshold > 0 ) - alienNextStageThreshold = ceil( (float)alienNextStageThreshold / 100 ) * 100; - - if( humanNextStageThreshold > 0 ) - humanNextStageThreshold = ceil( (float)humanNextStageThreshold / 100 ) * 100; - - trap_SetConfigstring( CS_ALIEN_STAGES, va( "%d %d %d", - g_alienStage.integer, g_alienCredits.integer, - alienNextStageThreshold ) ); - - trap_SetConfigstring( CS_HUMAN_STAGES, va( "%d %d %d", - g_humanStage.integer, g_humanCredits.integer, - humanNextStageThreshold ) ); -} - -/* -============ -CalculateAvgPlayers - -Calculates the average number of players playing this game -============ -*/ -void G_CalculateAvgPlayers( void ) -{ - //there are no clients or only spectators connected, so - //reset the number of samples in order to avoid the situation - //where the average tends to 0 - if( !level.numAlienClients ) - { - level.numAlienSamples = 0; - trap_Cvar_Set( "g_alienCredits", "0" ); - } - - if( !level.numHumanClients ) - { - level.numHumanSamples = 0; - trap_Cvar_Set( "g_humanCredits", "0" ); - } - - //calculate average number of clients for stats - level.averageNumAlienClients = - ( ( level.averageNumAlienClients * level.numAlienSamples ) - + level.numAlienClients ) / - (float)( level.numAlienSamples + 1 ); - level.numAlienSamples++; - - level.averageNumHumanClients = - ( ( level.averageNumHumanClients * level.numHumanSamples ) - + level.numHumanClients ) / - (float)( level.numHumanSamples + 1 ); - level.numHumanSamples++; -} - -/* -============ -CalculateRanks - -Recalculates the score ranks of all players -This will be called on every client connect, begin, disconnect, death, -and team change. -============ -*/ -void CalculateRanks( void ) -{ - int i; - char P[ MAX_CLIENTS + 1 ] = {""}; - - level.numConnectedClients = 0; - level.numPlayingClients = 0; - memset( level.numVotingClients, 0, sizeof( level.numVotingClients ) ); - level.numAlienClients = 0; - level.numHumanClients = 0; - level.numLiveAlienClients = 0; - level.numLiveHumanClients = 0; - - for( i = 0; i < level.maxclients; i++ ) - { - P[ i ] = '-'; - if ( level.clients[ i ].pers.connected != CON_DISCONNECTED ) - { - level.sortedClients[ level.numConnectedClients ] = i; - level.numConnectedClients++; - P[ i ] = (char)'0' + level.clients[ i ].pers.teamSelection; - - level.numVotingClients[ TEAM_NONE ]++; - - if( level.clients[ i ].pers.connected != CON_CONNECTED ) - continue; - - if( level.clients[ i ].pers.teamSelection != TEAM_NONE ) - { - level.numPlayingClients++; - if( level.clients[ i ].pers.teamSelection == TEAM_ALIENS ) - { - level.numAlienClients++; - if( level.clients[ i ].sess.spectatorState == SPECTATOR_NOT ) - level.numLiveAlienClients++; - } - else if( level.clients[ i ].pers.teamSelection == TEAM_HUMANS ) - { - level.numHumanClients++; - if( level.clients[ i ].sess.spectatorState == SPECTATOR_NOT ) - level.numLiveHumanClients++; - } - } - } - } - level.numNonSpectatorClients = level.numLiveAlienClients + - level.numLiveHumanClients; - level.numVotingClients[ TEAM_ALIENS ] = level.numAlienClients; - level.numVotingClients[ TEAM_HUMANS ] = level.numHumanClients; - P[ i ] = '\0'; - trap_Cvar_Set( "P", P ); - - qsort( level.sortedClients, level.numConnectedClients, - sizeof( level.sortedClients[ 0 ] ), SortRanks ); - - // see if it is time to end the level - CheckExitRules( ); - - // if we are at the intermission, send the new info to everyone - if( level.intermissiontime ) - SendScoreboardMessageToAllClients( ); -} - - -/* -======================================================================== - -MAP CHANGING - -======================================================================== -*/ - -/* -======================== -SendScoreboardMessageToAllClients - -Do this at BeginIntermission time and whenever ranks are recalculated -due to enters/exits/forced team changes -======================== -*/ -void SendScoreboardMessageToAllClients( void ) -{ - int i; - - for( i = 0; i < level.maxclients; i++ ) - { - if( level.clients[ i ].pers.connected == CON_CONNECTED ) - ScoreboardMessage( g_entities + i ); - } -} - -/* -======================== -MoveClientToIntermission - -When the intermission starts, this will be called for all players. -If a new client connects, this will be called after the spawn function. -======================== -*/ -void MoveClientToIntermission( gentity_t *ent ) -{ - // take out of follow mode if needed - if( ent->client->sess.spectatorState == SPECTATOR_FOLLOW ) - G_StopFollowing( ent ); - - // move to the spot - VectorCopy( level.intermission_origin, ent->s.origin ); - VectorCopy( level.intermission_origin, ent->client->ps.origin ); - VectorCopy( level.intermission_angle, ent->client->ps.viewangles ); - ent->client->ps.pm_type = PM_INTERMISSION; - - // clean up powerup info - memset( ent->client->ps.misc, 0, sizeof( ent->client->ps.misc ) ); - - ent->client->ps.eFlags = 0; - ent->s.eFlags = 0; - ent->s.eType = ET_GENERAL; - ent->s.modelindex = 0; - ent->s.loopSound = 0; - ent->s.event = 0; - ent->r.contents = 0; -} - -/* -================== -FindIntermissionPoint - -This is also used for spectator spawns -================== -*/ -void FindIntermissionPoint( void ) -{ - gentity_t *ent, *target; - vec3_t dir; - - // find the intermission spot - ent = G_Find( NULL, FOFS( classname ), "info_player_intermission" ); - - if( !ent ) - { // the map creator forgot to put in an intermission point... - G_SelectSpawnPoint( vec3_origin, level.intermission_origin, level.intermission_angle ); - } - else - { - VectorCopy( ent->s.origin, level.intermission_origin ); - VectorCopy( ent->s.angles, level.intermission_angle ); - // if it has a target, look towards it - if( ent->target ) - { - target = G_PickTarget( ent->target ); - - if( target ) - { - VectorSubtract( target->s.origin, level.intermission_origin, dir ); - vectoangles( dir, level.intermission_angle ); - } - } - } - -} - -/* -================== -BeginIntermission -================== -*/ -void BeginIntermission( void ) -{ - int i; - gentity_t *client; - - if( level.intermissiontime ) - return; // already active - - level.intermissiontime = level.time; - - G_ClearVotes( ); - - G_UpdateTeamConfigStrings( ); - - FindIntermissionPoint( ); - - // move all clients to the intermission point - for( i = 0; i < level.maxclients; i++ ) - { - client = g_entities + i; - - if( !client->inuse ) - continue; - - // respawn if dead - if( client->health <= 0 ) - respawn(client); - - MoveClientToIntermission( client ); - } - - // send the current scoring to all clients - SendScoreboardMessageToAllClients( ); -} - - -/* -============= -ExitLevel - -When the intermission has been exited, the server is either moved -to a new map based on the map rotation or the current map restarted -============= -*/ -void ExitLevel( void ) -{ - int i; - gclient_t *cl; - - if ( G_MapExists( g_nextMap.string ) ) - trap_SendConsoleCommand( EXEC_APPEND, va("map \"%s\"\n", g_nextMap.string ) ); - else if( G_MapRotationActive( ) ) - G_AdvanceMapRotation( 0 ); - else - trap_SendConsoleCommand( EXEC_APPEND, "map_restart\n" ); - - trap_Cvar_Set( "g_nextMap", "" ); - - level.restarted = qtrue; - level.changemap = NULL; - level.intermissiontime = 0; - - // reset all the scores so we don't enter the intermission again - for( i = 0; i < g_maxclients.integer; i++ ) - { - cl = level.clients + i; - if( cl->pers.connected != CON_CONNECTED ) - continue; - - cl->ps.persistant[ PERS_SCORE ] = 0; - } - - // we need to do this here before chaning to CON_CONNECTING - G_WriteSessionData( ); - - // change all client states to connecting, so the early players into the - // next level will know the others aren't done reconnecting - for( i = 0; i < g_maxclients.integer; i++ ) - { - if( level.clients[ i ].pers.connected == CON_CONNECTED ) - level.clients[ i ].pers.connected = CON_CONNECTING; - } - -} - -/* -================= -G_AdminMessage - -Print to all active server admins, and to the logfile, and to the server console -================= -*/ -void G_AdminMessage( gentity_t *ent, const char *msg ) -{ - char string[ 1024 ]; - int i; - - Com_sprintf( string, sizeof( string ), "chat %d %d \"%s\"", - ent ? ent - g_entities : -1, - G_admin_permission( ent, ADMF_ADMINCHAT ) ? SAY_ADMINS : SAY_ADMINS_PUBLIC, - msg ); - - // Send to all appropriate clients - for( i = 0; i < level.maxclients; i++ ) - if( G_admin_permission( &g_entities[ i ], ADMF_ADMINCHAT ) ) - trap_SendServerCommand( i, string ); - - // Send to the logfile and server console - G_LogPrintf( "%s: %d \"%s" S_COLOR_WHITE "\": " S_COLOR_MAGENTA "%s\n", - G_admin_permission( ent, ADMF_ADMINCHAT ) ? "AdminMsg" : "AdminMsgPublic", - ent ? ent - g_entities : -1, ent ? ent->client->pers.netname : "console", - msg ); -} - - -/* -================= -G_LogPrintf - -Print to the logfile with a time stamp if it is open, and to the server console -================= -*/ -void QDECL G_LogPrintf( const char *fmt, ... ) -{ - va_list argptr; - char string[ 1024 ], decolored[ 1024 ]; - int min, tens, sec; - - sec = ( level.time - level.startTime ) / 1000; - - min = sec / 60; - sec -= min * 60; - tens = sec / 10; - sec -= tens * 10; - - Com_sprintf( string, sizeof( string ), "%3i:%i%i ", min, tens, sec ); - - va_start( argptr, fmt ); - Q_vsnprintf( string + 7, sizeof( string ) - 7, fmt, argptr ); - va_end( argptr ); - - if( g_dedicated.integer ) - { - G_UnEscapeString( string, decolored, sizeof( decolored ) ); - G_Printf( "%s", decolored + 7 ); - } - - if( !level.logFile ) - return; - - G_DecolorString( string, decolored, sizeof( decolored ) ); - trap_FS_Write( decolored, strlen( decolored ), level.logFile ); -} - -/* -================= -G_SendGameStat -================= -*/ -void G_SendGameStat( team_t team ) -{ - char map[ MAX_STRING_CHARS ]; - char teamChar; - char data[ BIG_INFO_STRING ]; - char entry[ MAX_STRING_CHARS ]; - int i, dataLength, entryLength; - gclient_t *cl; - - // games with cheats enabled are not very good for balance statistics - if( g_cheats.integer ) - return; - - trap_Cvar_VariableStringBuffer( "mapname", map, sizeof( map ) ); - - switch( team ) - { - case TEAM_ALIENS: teamChar = 'A'; break; - case TEAM_HUMANS: teamChar = 'H'; break; - case TEAM_NONE: teamChar = 'L'; break; - default: return; - } - - Com_sprintf( data, BIG_INFO_STRING, - "%s %s T:%c A:%f H:%f M:%s D:%d SD:%d AS:%d AS2T:%d AS3T:%d HS:%d HS2T:%d HS3T:%d CL:%d", - Q3_VERSION, - g_tag.string, - teamChar, - level.averageNumAlienClients, - level.averageNumHumanClients, - map, - level.time - level.startTime, - G_TimeTilSuddenDeath( ), - g_alienStage.integer, - level.alienStage2Time - level.startTime, - level.alienStage3Time - level.startTime, - g_humanStage.integer, - level.humanStage2Time - level.startTime, - level.humanStage3Time - level.startTime, - level.numConnectedClients ); - - dataLength = strlen( data ); - - for( i = 0; i < level.numConnectedClients; i++ ) - { - int ping; - - cl = &level.clients[ level.sortedClients[ i ] ]; - - if( cl->pers.connected == CON_CONNECTING ) - ping = -1; - else - ping = cl->ps.ping < 999 ? cl->ps.ping : 999; - - switch( cl->ps.stats[ STAT_TEAM ] ) - { - case TEAM_ALIENS: teamChar = 'A'; break; - case TEAM_HUMANS: teamChar = 'H'; break; - case TEAM_NONE: teamChar = 'S'; break; - default: return; - } - - Com_sprintf( entry, MAX_STRING_CHARS, - " \"%s\" %c %d %d %d", - cl->pers.netname, - teamChar, - cl->ps.persistant[ PERS_SCORE ], - ping, - ( level.time - cl->pers.enterTime ) / 60000 ); - - entryLength = strlen( entry ); - - if( dataLength + entryLength >= BIG_INFO_STRING ) - break; - - strcpy( data + dataLength, entry ); - dataLength += entryLength; - } - - trap_SendGameStat( data ); -} - -/* -================ -LogExit - -Append information about this game to the log file -================ -*/ -void LogExit( const char *string ) -{ - int i, numSorted; - gclient_t *cl; - gentity_t *ent; - - G_LogPrintf( "Exit: %s\n", string ); - - level.intermissionQueued = level.time; - - // this will keep the clients from playing any voice sounds - // that will get cut off when the queued intermission starts - trap_SetConfigstring( CS_INTERMISSION, "1" ); - - // don't send more than 32 scores (FIXME?) - numSorted = level.numConnectedClients; - if( numSorted > 32 ) - numSorted = 32; - - for( i = 0; i < numSorted; i++ ) - { - int ping; - - cl = &level.clients[ level.sortedClients[ i ] ]; - - if( cl->ps.stats[ STAT_TEAM ] == TEAM_NONE ) - continue; - - if( cl->pers.connected == CON_CONNECTING ) - continue; - - ping = cl->ps.ping < 999 ? cl->ps.ping : 999; - - G_LogPrintf( "score: %i ping: %i client: %i %s\n", - cl->ps.persistant[ PERS_SCORE ], ping, level.sortedClients[ i ], - cl->pers.netname ); - - } - - for( i = 1, ent = g_entities + i ; i < level.num_entities ; i++, ent++ ) - { - if( !ent->inuse ) - continue; - - if( !Q_stricmp( ent->classname, "trigger_win" ) ) - { - if( level.lastWin == ent->stageTeam ) - ent->use( ent, ent, ent ); - } - } - - G_SendGameStat( level.lastWin ); -} - - -/* -================= -CheckIntermissionExit - -The level will stay at the intermission for a minimum of 5 seconds -If all players wish to continue, the level will then exit. -If one or more players have not acknowledged the continue, the game will -wait 10 seconds before going on. -================= -*/ -void CheckIntermissionExit( void ) -{ - int ready, notReady; - int i; - gclient_t *cl; - clientList_t readyMasks; - - //if no clients are connected, just exit - if( level.numConnectedClients == 0 ) - { - ExitLevel( ); - return; - } - - // see which players are ready - ready = 0; - notReady = 0; - Com_Memset( &readyMasks, 0, sizeof( readyMasks ) ); - for( i = 0; i < g_maxclients.integer; i++ ) - { - cl = level.clients + i; - - if( cl->pers.connected != CON_CONNECTED ) - continue; - - if( cl->ps.stats[ STAT_TEAM ] == TEAM_NONE ) - continue; - - if( cl->readyToExit ) - { - ready++; - - Com_ClientListAdd( &readyMasks, i ); - } - else - notReady++; - } - - trap_SetConfigstring( CS_CLIENTS_READY, Com_ClientListString( &readyMasks ) ); - - // never exit in less than five seconds - if( level.time < level.intermissiontime + 5000 ) - return; - - // never let intermission go on for over 1 minute - if( level.time > level.intermissiontime + 60000 ) - { - ExitLevel( ); - return; - } - - // if nobody wants to go, clear timer - if( ready == 0 && notReady > 0 ) - { - level.readyToExit = qfalse; - return; - } - - // if everyone wants to go, go now - if( notReady == 0 ) - { - ExitLevel( ); - return; - } - - // the first person to ready starts the thirty second timeout - if( !level.readyToExit ) - { - level.readyToExit = qtrue; - level.exitTime = level.time; - } - - // if we have waited thirty seconds since at least one player - // wanted to exit, go ahead - if( level.time < level.exitTime + 30000 ) - return; - - ExitLevel( ); -} - -/* -============= -ScoreIsTied -============= -*/ -qboolean ScoreIsTied( void ) -{ - int a, b; - - if( level.numPlayingClients < 2 ) - return qfalse; - - a = level.clients[ level.sortedClients[ 0 ] ].ps.persistant[ PERS_SCORE ]; - b = level.clients[ level.sortedClients[ 1 ] ].ps.persistant[ PERS_SCORE ]; - - return a == b; -} - -/* -================= -CheckExitRules - -There will be a delay between the time the exit is qualified for -and the time everyone is moved to the intermission spot, so you -can see the last frag. -================= -*/ -void CheckExitRules( void ) -{ - // if at the intermission, wait for all non-bots to - // signal ready, then go to next level - if( level.intermissiontime ) - { - CheckIntermissionExit( ); - return; - } - - if( level.intermissionQueued ) - { - if( level.time - level.intermissionQueued >= INTERMISSION_DELAY_TIME ) - { - level.intermissionQueued = 0; - BeginIntermission( ); - } - - return; - } - - if( g_timelimit.integer ) - { - if( level.time - level.startTime >= g_timelimit.integer * 60000 ) - { - level.lastWin = TEAM_NONE; - trap_SendServerCommand( -1, "print \"Timelimit hit\n\"" ); - trap_SetConfigstring( CS_WINNER, "Stalemate" ); - LogExit( "Timelimit hit." ); - return; - } - else if( level.time - level.startTime >= ( g_timelimit.integer - 5 ) * 60000 && - level.timelimitWarning < TW_IMMINENT ) - { - trap_SendServerCommand( -1, "cp \"5 minutes remaining!\"" ); - level.timelimitWarning = TW_IMMINENT; - } - else if( level.time - level.startTime >= ( g_timelimit.integer - 1 ) * 60000 && - level.timelimitWarning < TW_PASSED ) - { - trap_SendServerCommand( -1, "cp \"1 minute remaining!\"" ); - level.timelimitWarning = TW_PASSED; - } - } - - if( level.uncondHumanWin || - ( !level.uncondAlienWin && - ( level.time > level.startTime + 1000 ) && - ( level.numAlienSpawns == 0 ) && - ( level.numLiveAlienClients == 0 ) ) ) - { - //humans win - level.lastWin = TEAM_HUMANS; - trap_SendServerCommand( -1, "print \"Humans win\n\""); - trap_SetConfigstring( CS_WINNER, "Humans Win" ); - LogExit( "Humans win." ); - } - else if( level.uncondAlienWin || - ( ( level.time > level.startTime + 1000 ) && - ( level.numHumanSpawns == 0 ) && - ( level.numLiveHumanClients == 0 ) ) ) - { - //aliens win - level.lastWin = TEAM_ALIENS; - trap_SendServerCommand( -1, "print \"Aliens win\n\""); - trap_SetConfigstring( CS_WINNER, "Aliens Win" ); - LogExit( "Aliens win." ); - } -} - -/* -================== -G_Vote -================== -*/ -void G_Vote( gentity_t *ent, team_t team, qboolean voting ) -{ - if( !level.voteTime[ team ] ) - return; - - if( voting && ent->client->pers.voted & ( 1 << team ) ) - return; - - if( !voting && !( ent->client->pers.voted & ( 1 << team ) ) ) - return; - - ent->client->pers.voted |= 1 << team; - - if( ent->client->pers.vote & ( 1 << team ) ) - { - if( voting ) - level.voteYes[ team ]++; - else - level.voteYes[ team ]--; - - trap_SetConfigstring( CS_VOTE_YES + team, - va( "%d", level.voteYes[ team ] ) ); - } - else - { - if( voting ) - level.voteNo[ team ]++; - else - level.voteNo[ team ]--; - - trap_SetConfigstring( CS_VOTE_NO + team, - va( "%d", level.voteNo[ team ] ) ); - } -} - - -/* -======================================================================== - -FUNCTIONS CALLED EVERY FRAME - -======================================================================== -*/ - - -void G_ExecuteVote( team_t team ) -{ - level.voteExecuteTime[ team ] = 0; - - trap_SendConsoleCommand( EXEC_APPEND, va( "%s\n", - level.voteString[ team ] ) ); - - if( !Q_stricmpn( level.voteString[ team ], "map", 3 ) ) - level.restarted = qtrue; -} - -/* -================== -G_CheckVote -================== -*/ -void G_CheckVote( team_t team ) -{ - float votePassThreshold = (float)level.voteThreshold[ team ] / 100.0f; - qboolean pass = qfalse; - char *msg; - int i; - - if( level.voteExecuteTime[ team ] && - level.voteExecuteTime[ team ] < level.time ) - { - G_ExecuteVote( team ); - } - - if( !level.voteTime[ team ] ) - return; - - if( ( level.time - level.voteTime[ team ] >= VOTE_TIME ) || - ( level.voteYes[ team ] + level.voteNo[ team ] == level.numVotingClients[ team ] ) ) - { - pass = ( level.voteYes[ team ] && - (float)level.voteYes[ team ] / ( (float)level.voteYes[ team ] + (float)level.voteNo[ team ] ) > votePassThreshold ); - } - else - { - if( (float)level.voteYes[ team ] > - (float)level.numVotingClients[ team ] * votePassThreshold ) - { - pass = qtrue; - } - else if( (float)level.voteNo[ team ] <= - (float)level.numVotingClients[ team ] * ( 1.0f - votePassThreshold ) ) - { - return; - } - } - - if( pass ) - level.voteExecuteTime[ team ] = level.time + level.voteDelay[ team ]; - - G_LogPrintf( "EndVote: %s %s %d %d %d\n", - team == TEAM_NONE ? "global" : BG_TeamName( team ), - pass ? "pass" : "fail", - level.voteYes[ team ], level.voteNo[ team ], level.numVotingClients[ team ] ); - - msg = va( "print \"%sote %sed (%d - %d)\n\"", - team == TEAM_NONE ? "V" : "Team v", pass ? "pass" : "fail", - level.voteYes[ team ], level.voteNo[ team ] ); - - if( team == TEAM_NONE ) - trap_SendServerCommand( -1, msg ); - else - G_TeamCommand( team, msg ); - - level.voteTime[ team ] = 0; - level.voteYes[ team ] = 0; - level.voteNo[ team ] = 0; - - for( i = 0; i < level.maxclients; i++ ) - level.clients[ i ].pers.voted &= ~( 1 << team ); - - trap_SetConfigstring( CS_VOTE_TIME + team, "" ); - trap_SetConfigstring( CS_VOTE_STRING + team, "" ); - trap_SetConfigstring( CS_VOTE_YES + team, "0" ); - trap_SetConfigstring( CS_VOTE_NO + team, "0" ); -} - - -/* -================== -CheckCvars -================== -*/ -void CheckCvars( void ) -{ - static int lastPasswordModCount = -1; - static int lastMarkDeconModCount = -1; - static int lastSDTimeModCount = -1; - static int lastNumZones = 0; - - if( g_password.modificationCount != lastPasswordModCount ) - { - lastPasswordModCount = g_password.modificationCount; - - if( *g_password.string && Q_stricmp( g_password.string, "none" ) ) - trap_Cvar_Set( "g_needpass", "1" ); - else - trap_Cvar_Set( "g_needpass", "0" ); - } - - // Unmark any structures for deconstruction when - // the server setting is changed - if( g_markDeconstruct.modificationCount != lastMarkDeconModCount ) - { - lastMarkDeconModCount = g_markDeconstruct.modificationCount; - G_ClearDeconMarks( ); - } - - // If we change g_suddenDeathTime during a map, we need to update - // when sd will begin - if( g_suddenDeathTime.modificationCount != lastSDTimeModCount ) - { - lastSDTimeModCount = g_suddenDeathTime.modificationCount; - level.suddenDeathBeginTime = g_suddenDeathTime.integer * 60000; - } - - // If the number of zones changes, we need a new array - if( g_humanRepeaterMaxZones.integer != lastNumZones ) - { - buildPointZone_t *newZones; - size_t newsize = g_humanRepeaterMaxZones.integer * sizeof( buildPointZone_t ); - size_t oldsize = lastNumZones * sizeof( buildPointZone_t ); - - newZones = BG_Alloc( newsize ); - if( level.buildPointZones ) - { - Com_Memcpy( newZones, level.buildPointZones, MIN( oldsize, newsize ) ); - BG_Free( level.buildPointZones ); - } - - level.buildPointZones = newZones; - lastNumZones = g_humanRepeaterMaxZones.integer; - } - - level.frameMsec = trap_Milliseconds( ); -} - -/* -============= -G_RunThink - -Runs thinking code for this frame if necessary -============= -*/ -void G_RunThink( gentity_t *ent ) -{ - float thinktime; - - thinktime = ent->nextthink; - if( thinktime <= 0 ) - return; - - if( thinktime > level.time ) - return; - - ent->nextthink = 0; - if( !ent->think ) - G_Error( "NULL ent->think" ); - - ent->think( ent ); -} - -/* -============= -G_EvaluateAcceleration - -Calculates the acceleration for an entity -============= -*/ -void G_EvaluateAcceleration( gentity_t *ent, int msec ) -{ - vec3_t deltaVelocity; - vec3_t deltaAccel; - - VectorSubtract( ent->s.pos.trDelta, ent->oldVelocity, deltaVelocity ); - VectorScale( deltaVelocity, 1.0f / (float)msec, ent->acceleration ); - - VectorSubtract( ent->acceleration, ent->oldAccel, deltaAccel ); - VectorScale( deltaAccel, 1.0f / (float)msec, ent->jerk ); - - VectorCopy( ent->s.pos.trDelta, ent->oldVelocity ); - VectorCopy( ent->acceleration, ent->oldAccel ); -} - -/* -================ -G_RunFrame - -Advances the non-player objects in the world -================ -*/ -void G_RunFrame( int levelTime ) -{ - int i; - gentity_t *ent; - int msec; - static int ptime3000 = 0; - - // if we are waiting for the level to restart, do nothing - if( level.restarted ) - return; - - if( level.pausedTime ) - { - msec = levelTime - level.time - level.pausedTime; - level.pausedTime = levelTime - level.time; - - ptime3000 += msec; - while( ptime3000 > 3000 ) - { - ptime3000 -= 3000; - trap_SendServerCommand( -1, "cp \"The game has been paused. Please wait.\"" ); - - if( level.pausedTime >= 110000 && level.pausedTime <= 119000 ) - trap_SendServerCommand( -1, va( "print \"Server: Game will auto-unpause in %d seconds\n\"", - (int) ( (float) ( 120000 - level.pausedTime ) / 1000.0f ) ) ); - } - - // Prevents clients from getting lagged-out messages - for( i = 0; i < level.maxclients; i++ ) - { - if( level.clients[ i ].pers.connected == CON_CONNECTED ) - level.clients[ i ].ps.commandTime = levelTime; - } - - if( level.pausedTime > 120000 ) - { - trap_SendServerCommand( -1, "print \"Server: The game has been unpaused automatically (2 minute max)\n\"" ); - trap_SendServerCommand( -1, "cp \"The game has been unpaused!\"" ); - level.pausedTime = 0; - } - - return; - } - - level.framenum++; - level.previousTime = level.time; - level.time = levelTime; - msec = level.time - level.previousTime; - - // get any cvar changes - G_UpdateCvars( ); - CheckCvars( ); - // now we are done spawning - level.spawning = qfalse; - - // - // go through all allocated objects - // - ent = &g_entities[ 0 ]; - - for( i = 0; i < level.num_entities; i++, ent++ ) - { - if( !ent->inuse ) - continue; - - // clear events that are too old - if( level.time - ent->eventTime > EVENT_VALID_MSEC ) - { - if( ent->s.event ) - { - ent->s.event = 0; // &= EV_EVENT_BITS; - if ( ent->client ) - { - ent->client->ps.externalEvent = 0; - //ent->client->ps.events[0] = 0; - //ent->client->ps.events[1] = 0; - } - } - - if( ent->freeAfterEvent ) - { - // tempEntities or dropped items completely go away after their event - G_FreeEntity( ent ); - continue; - } - else if( ent->unlinkAfterEvent ) - { - // items that will respawn will hide themselves after their pickup event - ent->unlinkAfterEvent = qfalse; - trap_UnlinkEntity( ent ); - } - } - - // temporary entities don't think - if( ent->freeAfterEvent ) - continue; - - // calculate the acceleration of this entity - if( ent->evaluateAcceleration ) - G_EvaluateAcceleration( ent, msec ); - - if( !ent->r.linked && ent->neverFree ) - continue; - - if( ent->s.eType == ET_MISSILE ) - { - G_RunMissile( ent ); - continue; - } - - if( ent->s.eType == ET_BUILDABLE ) - { - G_BuildableThink( ent, msec ); - continue; - } - - if( ent->s.eType == ET_CORPSE || ent->physicsObject ) - { - G_Physics( ent, msec ); - continue; - } - - if( ent->s.eType == ET_MOVER ) - { - G_RunMover( ent ); - continue; - } - - if( i < MAX_CLIENTS ) - { - G_RunClient( ent ); - continue; - } - - G_RunThink( ent ); - } - - // perform final fixups on the players - ent = &g_entities[ 0 ]; - - for( i = 0; i < level.maxclients; i++, ent++ ) - { - if( ent->inuse ) - ClientEndFrame( ent ); - } - - // save position information for all active clients - G_UnlaggedStore( ); - - G_CountSpawns( ); - G_CalculateBuildPoints( ); - G_CalculateStages( ); - - G_SpawnClients( TEAM_ALIENS ); - G_SpawnClients( TEAM_HUMANS ); - G_CalculateAvgPlayers( ); - G_UpdateZaps( msec ); - - // see if it is time to end the level - CheckExitRules( ); - - // update to team status? - CheckTeamStatus( ); - - // cancel vote if timed out - for( i = 0; i < NUM_TEAMS; i++ ) - G_CheckVote( i ); - - level.frameMsec = trap_Milliseconds(); -} - |