mirror of
https://github.com/ValveSoftware/halflife.git
synced 2025-01-27 22:17:59 +03:00
521 lines
11 KiB
C++
521 lines
11 KiB
C++
//========= Copyright © 1996-2001, Valve LLC, All rights reserved. ============
|
|
//
|
|
// Purpose:
|
|
//
|
|
// $NoKeywords: $
|
|
//=============================================================================
|
|
|
|
#include "extdll.h"
|
|
#include "entity_state.h"
|
|
#include "pm_defs.h"
|
|
#include "pm_shared.h"
|
|
#include "pm_movevars.h"
|
|
#include "hud_iface.h"
|
|
#include "com_model.h"
|
|
#include "event_api.h"
|
|
#include "com_weapons.h"
|
|
#include "event_flags.h"
|
|
#include "DMC_BSPFile.h"
|
|
#include "cl_util.h"
|
|
|
|
#include "FileSystem.h"
|
|
|
|
extern IFileSystem *g_pFileSystem;
|
|
|
|
extern "C" playermove_t *pmove;
|
|
extern int g_runfuncs;
|
|
|
|
// Don't support more than MAX_TELE teleporters ( map still can load tho )
|
|
#define MAX_TELES 256
|
|
|
|
extern Vector g_vecTeleMins[ MAX_TELES ];
|
|
extern Vector g_vecTeleMaxs[ MAX_TELES ];
|
|
extern int g_iTeleNum;
|
|
extern int g_iUser1;
|
|
extern bool g_bLoadedTeles;
|
|
vec3_t vecTempAngles;
|
|
bool bChangeAngles;
|
|
|
|
|
|
// We only care about two kinds of entities for now: Teleporters and their targets
|
|
// FIXME: After loading, store a pointer from teleporter to target instead of looking up all the time.
|
|
typedef enum
|
|
{
|
|
// Entity is a teleporter
|
|
DMC_TELE = 0,
|
|
// Entity is a target
|
|
DMC_TARGET
|
|
} dmc_teletype_t;
|
|
|
|
typedef struct
|
|
{
|
|
// Type of entity
|
|
dmc_teletype_t type;
|
|
|
|
// Classname
|
|
char classname[ 32 ];
|
|
|
|
// What this entity targets
|
|
char target[ 32 ];
|
|
|
|
// If entity is a target, the name tag it uses
|
|
char targetname[ 32 ];
|
|
|
|
// Orientation of the teleporter
|
|
float angles[3];
|
|
|
|
// Target origin
|
|
float origin[3];
|
|
|
|
// Bounding box of the teleporter
|
|
float absmin[3];
|
|
float absmax[3];
|
|
|
|
} dmc_tele_t;
|
|
|
|
// Teleporter/Target entity database
|
|
static dmc_tele_t s_teles[ MAX_TELES ];
|
|
static int s_num_teles = 0;
|
|
|
|
// We'll use this for playing the teleporting sounds locally.
|
|
static unsigned short s_usTeleport;
|
|
|
|
/*
|
|
==============================
|
|
Dmc_SetKeyValue
|
|
|
|
Fill in key/values fro the teleporter
|
|
==============================
|
|
*/
|
|
void Dmc_SetKeyValue( dmc_tele_t *pTele, const char *key, const char *value )
|
|
{
|
|
float x, y, z;
|
|
|
|
if ( !stricmp( key, "classname" ) )
|
|
{
|
|
strcpy( pTele->classname, value );
|
|
}
|
|
else if ( !stricmp( key, "target" ) )
|
|
{
|
|
strcpy( pTele->target, value );
|
|
}
|
|
else if ( !stricmp( key, "targetname" ) )
|
|
{
|
|
strcpy( pTele->targetname, value );
|
|
}
|
|
else if ( !stricmp( key, "angles" ) )
|
|
{
|
|
if ( sscanf( value, "%f %f %f", &x, &y, &z ) == 3 )
|
|
{
|
|
pTele->angles[ 0 ] = x ;
|
|
pTele->angles[ 1 ] = y;
|
|
pTele->angles[ 2 ] = z;
|
|
}
|
|
}
|
|
else if ( !stricmp( key, "origin" ) )
|
|
{
|
|
if ( sscanf( value, "%f %f %f", &x, &y, &z ) == 3 )
|
|
{
|
|
pTele->origin[ 0 ] = x;
|
|
pTele->origin[ 1 ] = y;
|
|
pTele->origin[ 2 ] = z;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
==============================
|
|
Dmc_ParseTeleporter
|
|
|
|
Evaluate Key/Value pairs for the entity
|
|
==============================
|
|
*/
|
|
char *Dmc_ParseTeleporter( char *buffer, dmc_tele_t *pTele, int *error )
|
|
{
|
|
char key[256];
|
|
char token[ 1024 ];
|
|
int n;
|
|
|
|
memset( pTele, 0, sizeof( *pTele ) );
|
|
|
|
while (1)
|
|
{
|
|
// Parse key
|
|
buffer = gEngfuncs.COM_ParseFile ( buffer, token );
|
|
if ( token[0] == '}' )
|
|
break;
|
|
|
|
// Ran out of input buffer?
|
|
if ( !buffer )
|
|
{
|
|
*error = 1;
|
|
break;
|
|
}
|
|
|
|
// Store off the key
|
|
strcpy ( key, token );
|
|
|
|
// Fix heynames with trailing spaces
|
|
n = strlen( key );
|
|
while (n && key[n-1] == ' ')
|
|
{
|
|
key[n-1] = 0;
|
|
n--;
|
|
}
|
|
|
|
// Parse value
|
|
buffer = gEngfuncs.COM_ParseFile ( buffer, token );
|
|
|
|
// Ran out of buffer?
|
|
if (!buffer)
|
|
{
|
|
*error = 1;
|
|
break;
|
|
}
|
|
|
|
// Hit the end instead of a value?
|
|
if ( token[0] == '}' )
|
|
{
|
|
*error = 1;
|
|
break;
|
|
}
|
|
|
|
if ( token[0] == '}' && token[1] == '(' )
|
|
int k = 0;
|
|
|
|
// Assign k/v pair
|
|
Dmc_SetKeyValue( pTele, key, token );
|
|
}
|
|
|
|
// Return what's left in the stream
|
|
return buffer;
|
|
}
|
|
|
|
/*
|
|
==============================
|
|
Dmc_ProcessEnts
|
|
|
|
Parse through entity lump looking for teleporters or targets
|
|
==============================
|
|
*/
|
|
void Dmc_ProcessEnts( char *buffer )
|
|
{
|
|
char token[ 1024 ];
|
|
dmc_tele_t *pTele = NULL;
|
|
int error = 0;
|
|
|
|
// parse entities from entity lump of .bsp file
|
|
while ( 1 )
|
|
{
|
|
// parse the opening brace
|
|
buffer = gEngfuncs.COM_ParseFile ( buffer, token );
|
|
if (!buffer)
|
|
break;
|
|
|
|
// Didn't find opening brace?
|
|
if ( token[0] != '{' )
|
|
{
|
|
gEngfuncs.Con_Printf ("Dmc_ProcessEnts: found %s when expecting {\n", token );
|
|
return;
|
|
}
|
|
|
|
// Assume we're filling in this tele
|
|
pTele = &s_teles[ s_num_teles ];
|
|
|
|
// Fill in data
|
|
buffer = Dmc_ParseTeleporter( buffer, pTele, &error );
|
|
|
|
// Check for errors and abort if any
|
|
if ( error )
|
|
{
|
|
gEngfuncs.Con_Printf ("Dmc_ProcessEnts: error parsing entities\n" );
|
|
return;
|
|
}
|
|
|
|
// Check classname
|
|
if ( stricmp( pTele->classname, "trigger_teleport" ) && stricmp( pTele->classname, "info_teleport_destination" ) )
|
|
continue;
|
|
|
|
// Set type based on classname
|
|
if ( !stricmp( pTele->classname, "trigger_teleport" ) )
|
|
{
|
|
pTele->type = DMC_TELE;
|
|
}
|
|
else
|
|
{
|
|
pTele->type = DMC_TARGET;
|
|
}
|
|
|
|
// If we got to here, we're using the entity
|
|
s_num_teles++;
|
|
|
|
// No more room...
|
|
if ( s_num_teles >= MAX_TELES )
|
|
break;
|
|
}
|
|
}
|
|
|
|
/*
|
|
==============================
|
|
Dmc_LoadEntityLump
|
|
|
|
Open the .bsp and read in the entity lump
|
|
==============================
|
|
*/
|
|
char *Dmc_LoadEntityLump( const char *filename )
|
|
{
|
|
FileHandle_t fp;
|
|
int i;
|
|
dheader_t header;
|
|
int size;
|
|
lump_t *curLump;
|
|
char *buffer = NULL;
|
|
|
|
fp = g_pFileSystem->Open( filename, "rb" );
|
|
if ( !fp )
|
|
return NULL;
|
|
|
|
// Read in the .bsp header
|
|
if ( g_pFileSystem->Read(&header, sizeof(dheader_t), fp) != sizeof(dheader_t) )
|
|
{
|
|
gEngfuncs.Con_Printf("Dmc_LoadEntityLump: Could not read BSP header for map [%s].\n", filename);
|
|
g_pFileSystem->Close(fp);
|
|
return NULL;
|
|
}
|
|
|
|
// Check the version
|
|
i = header.version;
|
|
if ( i != 29 && i != 30)
|
|
{
|
|
g_pFileSystem->Close(fp);
|
|
gEngfuncs.Con_Printf("Dmc_LoadEntityLump: Map [%s] has incorrect BSP version (%i should be %i).\n", filename, i, BSPVERSION);
|
|
return NULL;
|
|
}
|
|
|
|
// Get entity lump
|
|
curLump = &header.lumps[ LUMP_ENTITIES ];
|
|
// and entity lump size
|
|
size = curLump->filelen;
|
|
|
|
// Jump to it
|
|
g_pFileSystem->Seek( fp, curLump->fileofs, FILESYSTEM_SEEK_HEAD );
|
|
|
|
// Allocate sufficient memmory
|
|
buffer = (char *)malloc( size + 1 );
|
|
if ( !buffer )
|
|
{
|
|
g_pFileSystem->Close(fp);
|
|
gEngfuncs.Con_Printf("Dmc_LoadEntityLump: Couldn't allocate %i bytes\n", size + 1 );
|
|
return NULL;
|
|
}
|
|
|
|
// Read in the entity lump
|
|
g_pFileSystem->Read( buffer, size, fp );
|
|
|
|
// Terminate the string
|
|
buffer[ size ] = '\0';
|
|
|
|
if ( fp )
|
|
{
|
|
g_pFileSystem->Close(fp);
|
|
}
|
|
|
|
return buffer;
|
|
}
|
|
|
|
/*
|
|
==============================
|
|
Dmc_LoadTeleporters
|
|
|
|
Load in the .bsp file and process the entities
|
|
==============================
|
|
*/
|
|
void Dmc_LoadTeleporters( const char *map )
|
|
{
|
|
char *buffer = NULL;
|
|
char filename[ 256 ];
|
|
|
|
sprintf( filename, "%s", map );
|
|
|
|
// TODO: Fix Slashes?
|
|
|
|
// Reset count
|
|
s_num_teles = 0;
|
|
|
|
// Load entity lump
|
|
buffer = Dmc_LoadEntityLump( filename );
|
|
if ( !buffer )
|
|
return;
|
|
|
|
// Process buffer and extract teleporters/targets
|
|
Dmc_ProcessEnts( buffer );
|
|
|
|
// Discard buffer
|
|
free( buffer );
|
|
}
|
|
|
|
/*
|
|
==============================
|
|
Dmc_FindTarget
|
|
|
|
Search entity list for target matching "name"
|
|
==============================
|
|
*/
|
|
dmc_tele_t *Dmc_FindTarget( const char *name, int numtele, dmc_tele_t *pTeles )
|
|
{
|
|
int i;
|
|
dmc_tele_t *target;
|
|
|
|
// Find the target
|
|
for ( i = 0; i < numtele; i++ )
|
|
{
|
|
target = &pTeles[ i ];
|
|
|
|
if ( !target )
|
|
continue;
|
|
|
|
if ( stricmp( target->targetname, name ) )
|
|
continue;
|
|
|
|
return target;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
==============================
|
|
Dmc_TeleporterTouched
|
|
|
|
Imparts the desired velocity to the player
|
|
after touching a teleporter.
|
|
==============================
|
|
*/
|
|
void Dmc_TeleporterTouched( int numtele, dmc_tele_t *pTeles, dmc_tele_t *pTele, struct local_state_s *player )
|
|
{
|
|
int i;
|
|
dmc_tele_t *target;
|
|
pmtrace_t tr;
|
|
float flGravity = pmove->movevars->gravity;
|
|
|
|
vec3_t forward, up, right;
|
|
|
|
float zero[ 3 ] = { 0.0, 0.0, 0.0 };
|
|
|
|
// Find the target
|
|
target = Dmc_FindTarget( pTele->target, numtele, pTeles );
|
|
|
|
for ( i = 0; i < 3; i++ )
|
|
player->playerstate.origin[ i ] = target->origin[ i ];
|
|
|
|
player->playerstate.origin[ 2 ] += 27;
|
|
|
|
AngleVectors( target->angles, forward, right, up );
|
|
player->client.velocity = forward * 300;
|
|
|
|
// Play sound if appropriate
|
|
if ( s_usTeleport && g_runfuncs )
|
|
{
|
|
//Adrian - This is a little hack to make the player face
|
|
//the destination angles as soon as we step out.
|
|
//Check view.cpp for the rest.
|
|
for ( i = 0; i < 3; i++ )
|
|
vecTempAngles[ i ] = target->angles[ i ];
|
|
|
|
bChangeAngles = true;
|
|
|
|
gEngfuncs.pfnPlaybackEvent( FEV_NOTHOST, NULL, s_usTeleport, 0.0, target->origin, zero, 0.0, 0.0, 0, 0, 0, 0 );
|
|
}
|
|
}
|
|
|
|
/*
|
|
==============================
|
|
Dmc_TouchTeleporters
|
|
|
|
See if player is touching a teleporter ( not that kind of touching! ).
|
|
==============================
|
|
*/
|
|
void Dmc_TouchTeleporters ( struct local_state_s *player, dmc_tele_t *pTeles, int numtele )
|
|
{
|
|
int i, j;
|
|
dmc_tele_t *pTele;
|
|
float absmin[3], absmax[3];
|
|
float pmins[ 3 ] = { 13, 13, 24 };
|
|
float pmaxs[ 3 ] = { 13, 13, 32 };
|
|
vec3_t LengthVector;
|
|
int iTeleNum = 0;
|
|
|
|
|
|
// Determine player's bbox
|
|
for ( j = 0; j < 3; j++ )
|
|
{
|
|
absmin[ j ] = player->playerstate.origin[ j ] - pmins[ j ];
|
|
absmax[ j ] = player->playerstate.origin[ j ] + pmaxs[ j ];
|
|
}
|
|
|
|
for ( i = 0; i < numtele; i++ )
|
|
{
|
|
pTele = &pTeles[ i ];
|
|
if ( !pTele )
|
|
continue;
|
|
|
|
if ( pTele->type != DMC_TELE )
|
|
continue;
|
|
|
|
//Adrian - Load all the teleporter Mins and Max size.
|
|
//This comes via an event when the player connects.
|
|
if ( !g_bLoadedTeles )
|
|
{
|
|
for ( int j = 0; j < 3; j++ )
|
|
{
|
|
pTele->absmin[ j ] = g_vecTeleMins[ iTeleNum ][ j ] - 1.0;
|
|
pTele->absmax[ j ] = g_vecTeleMaxs[ iTeleNum ][ j ] + 1.0;
|
|
}
|
|
iTeleNum++;
|
|
|
|
//Done going thru all the teleporters
|
|
if ( iTeleNum == g_iTeleNum )
|
|
g_bLoadedTeles = true;
|
|
}
|
|
|
|
if ( absmin[0] > pTele->absmax[0]
|
|
|| absmin[1] > pTele->absmax[1]
|
|
|| absmin[2] > pTele->absmax[2]
|
|
|| absmax[0] < pTele->absmin[0]
|
|
|| absmax[1] < pTele->absmin[1]
|
|
|| absmax[2] < pTele->absmin[2] )
|
|
continue;
|
|
|
|
Dmc_TeleporterTouched( numtele, pTeles, pTele, player );
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
/*
|
|
==============================
|
|
Dmc_CheckTeleporters
|
|
|
|
Load data if needed, otherwise just run checks on player's final position to see if teleporter needs
|
|
to impart velocity on the player.
|
|
==============================
|
|
*/
|
|
void Dmc_CheckTeleporters( struct local_state_s *from, struct local_state_s *to )
|
|
{
|
|
static char current_level[ 128 ];
|
|
|
|
// See if we've changed to a new map
|
|
if ( stricmp( current_level, gEngfuncs.pfnGetLevelName() ) )
|
|
{
|
|
strcpy( current_level, gEngfuncs.pfnGetLevelName() );
|
|
Dmc_LoadTeleporters( current_level );
|
|
|
|
// Grab sound event
|
|
s_usTeleport = gEngfuncs.pfnPrecacheEvent( 1, "events/teleport.sc" );
|
|
}
|
|
|
|
// Run test, only if we're not a spectator
|
|
if ( g_iUser1 == OBS_NONE )
|
|
Dmc_TouchTeleporters( to, s_teles, s_num_teles );
|
|
}
|