reaction/reaction/cgame/cg_weapons.c

4054 lines
103 KiB
C
Raw Normal View History

//-----------------------------------------------------------------------------
//
// $Id$
//
//-----------------------------------------------------------------------------
//
2001-12-31 16:28:42 +00:00
// $Log$
2002-02-23 18:07:46 +00:00
// Revision 1.49 2002/02/23 18:07:46 slicer
// Changed Sniper code and Cam code
//
// Revision 1.48 2002/02/08 18:00:34 jbravo
// Fixing "No newline at end of file" Warnings Linux keeps giving me
//
// Revision 1.47 2002/01/14 01:19:23 niceass
// No more default 800 gravity on items - NiceAss
//
// Revision 1.46 2002/01/12 15:18:20 jbravo
// Added a fix from Niceass. (cg_drawgun affecting other player models)
//
// Revision 1.45 2002/01/11 19:48:29 jbravo
// Formatted the source in non DOS format.
//
2001-12-31 16:28:42 +00:00
// Revision 1.44 2001/12/31 16:28:41 jbravo
// I made a Booboo with the Log tag.
//
//
//-----------------------------------------------------------------------------
2001-07-28 12:32:54 +00:00
// Copyright (C) 1999-2000 Id Software, Inc.
2001-05-06 20:50:27 +00:00
//
// cg_weapons.c -- events and effects dealing with weapons
#include "cg_local.h"
/*
==========================
CG_ParseWeaponSoundFile
Added by Elder
Reads information for frame-sound timing
==========================
*/
static qboolean CG_ParseWeaponSoundFile( const char *filename, weaponInfo_t *weapon ) {
char *text_p;
int len;
int i;
char *token;
// float fps;
int skip; // Elder: What's this for?
char text[20000];
fileHandle_t f;
sfxWeapTiming_t *weapTiming;
weapTiming = weapon->animationSounds;
// load the file
len = trap_FS_FOpenFile( filename, &f, FS_READ );
if ( len <= 0 ) {
return qfalse;
}
if ( len >= sizeof( text ) - 1 ) {
CG_Printf( "File %s too long\n", filename );
return qfalse;
}
trap_FS_Read( text, len, f );
text[len] = 0;
trap_FS_FCloseFile( f );
// parse the text
text_p = text;
// Elder: uhh, what was this for?
skip = 0; // quite the compiler warning
for ( i = 0; i < MAX_ANIM_SOUNDS; i++ )
{
// Grab frame number
token = COM_Parse( &text_p );
if ( !token ) break;
// Add it to the array
if ( atoi( token ) )
{
2001-11-11 07:23:16 +00:00
//Com_Printf("(%i): %s\n", i, token);
weapTiming->sfxInfo[i].frame = atoi(token);
}
else
break;
//return qfalse;
// Grab sound file path
token = COM_Parse( &text_p );
if ( !token ) break;
//return qfalse;
2001-11-11 07:23:16 +00:00
//Com_Printf("(%i): %s\n", i, token);
weapTiming->sfxInfo[i].sound = trap_S_RegisterSound( token, qfalse );
}
// Store total number
weapTiming->numFrames = i;
2001-11-11 07:23:16 +00:00
//Com_Printf("Total Frames: %i\n", weapTiming->numFrames);
return qtrue;
}
2001-11-11 07:23:16 +00:00
2001-07-28 02:49:18 +00:00
/* [QUARANTINE] - Weapon Animations - CG_ParseWeaponAnimFile
==========================
CG_ParseWeaponAnimFile
==========================
*/
static qboolean CG_ParseWeaponAnimFile( const char *filename, weaponInfo_t *weapon )
2001-09-16 04:04:31 +00:00
{
2001-07-28 12:32:54 +00:00
char *text_p;
int len;
int i;
2001-09-16 04:04:31 +00:00
int skip;
2001-07-28 12:32:54 +00:00
char *token;
float fps;
char text[20000];
fileHandle_t f;
animation_t *animations;
animations = weapon->animations;
// load the file
len = trap_FS_FOpenFile( filename, &f, FS_READ );
if ( len <= 0 ) {
return qfalse;
}
if ( len >= sizeof( text ) - 1 ) {
CG_Printf( "File %s too long\n", filename );
return qfalse;
}
trap_FS_Read( text, len, f );
text[len] = 0;
trap_FS_FCloseFile( f );
2001-07-28 12:32:54 +00:00
// parse the text
text_p = text;
skip = 0; // quite the compiler warning
2001-07-28 12:32:54 +00:00
// read information for each frame
for ( i = 0 ; i < MAX_WEAPON_ANIMATIONS ; i++ ) {
2001-07-28 12:32:54 +00:00
token = COM_Parse( &text_p );
if ( !token ) break;
animations[i].firstFrame = atoi( token );
token = COM_Parse( &text_p );
if ( !token ) break;
animations[i].numFrames = atoi( token );
2001-07-28 12:32:54 +00:00
token = COM_Parse( &text_p );
if ( !token ) break;
animations[i].loopFrames = atoi( token );
2001-07-28 12:32:54 +00:00
token = COM_Parse( &text_p );
if ( !token ) break;
fps = atof( token );
if ( fps == 0 ) fps = 1;
animations[i].frameLerp = 1000 / fps;
animations[i].initialLerp = 1000 / fps;
}
if ( i != MAX_WEAPON_ANIMATIONS ) {
CG_Printf( "Error parsing weapon animation file: %s", filename );
return qfalse;
}
2001-07-28 02:49:18 +00:00
2001-07-28 12:32:54 +00:00
return qtrue;
2001-07-28 02:49:18 +00:00
}
// END
2001-05-06 20:50:27 +00:00
/*
==========================
CG_MachineGunEjectBrass
==========================
*/
static void CG_MachineGunEjectBrass( centity_t *cent ) {
localEntity_t *le;
refEntity_t *re;
vec3_t velocity, xvelocity;
vec3_t offset, xoffset;
float waterScale = 1.0f;
vec3_t v[3];
if ( cg_brassTime.integer <= 0 ) {
return;
}
le = CG_AllocLocalEntity();
re = &le->refEntity;
velocity[0] = 0;
velocity[1] = -50 + 40 * crandom();
velocity[2] = 100 + 50 * crandom();
le->leType = LE_FRAGMENT;
le->startTime = cg.time;
le->endTime = le->startTime + cg_brassTime.integer + ( cg_brassTime.integer / 4 ) * random();
le->pos.trType = TR_GRAVITY;
le->pos.trTime = cg.time - (rand()&15);
AnglesToAxis( cent->lerpAngles, v );
offset[0] = 8;
offset[1] = -4;
offset[2] = 24;
// NiceAss: Added for better starting location of brass
if (cg.predictedPlayerState.pm_flags & PMF_DUCKED) {
offset[2] -= 14;
}
2001-05-06 20:50:27 +00:00
xoffset[0] = offset[0] * v[0][0] + offset[1] * v[1][0] + offset[2] * v[2][0];
xoffset[1] = offset[0] * v[0][1] + offset[1] * v[1][1] + offset[2] * v[2][1];
xoffset[2] = offset[0] * v[0][2] + offset[1] * v[1][2] + offset[2] * v[2][2];
VectorAdd( cent->lerpOrigin, xoffset, re->origin );
VectorCopy( re->origin, le->pos.trBase );
if ( CG_PointContents( re->origin, -1 ) & CONTENTS_WATER ) {
waterScale = 0.10f;
}
xvelocity[0] = velocity[0] * v[0][0] + velocity[1] * v[1][0] + velocity[2] * v[2][0];
xvelocity[1] = velocity[0] * v[0][1] + velocity[1] * v[1][1] + velocity[2] * v[2][1];
xvelocity[2] = velocity[0] * v[0][2] + velocity[1] * v[1][2] + velocity[2] * v[2][2];
VectorScale( xvelocity, waterScale, le->pos.trDelta );
AxisCopy( axisDefault, re->axis );
re->hModel = cgs.media.machinegunBrassModel;
le->bounceFactor = 0.4 * waterScale;
le->angles.trType = TR_LINEAR;
le->angles.trTime = cg.time;
le->angles.trBase[0] = rand()&31;
le->angles.trBase[1] = rand()&31;
le->angles.trBase[2] = rand()&31;
le->angles.trDelta[0] = 2;
le->angles.trDelta[1] = 1;
le->angles.trDelta[2] = 0;
le->leFlags = LEF_TUMBLE;
le->leBounceSoundType = LEBS_BRASS;
le->leMarkType = LEMT_NONE;
}
/*
==========================
CG_ShotgunEjectBrass
==========================
*/
static void CG_ShotgunEjectBrass( centity_t *cent ) {
localEntity_t *le;
refEntity_t *re;
vec3_t velocity, xvelocity;
vec3_t offset, xoffset;
vec3_t v[3];
2001-11-11 07:23:16 +00:00
int i, isHC;
if (cent->currentState.weapon == WP_HANDCANNON)
isHC=1;
2001-05-06 20:50:27 +00:00
else
isHC=0;
if ( cg_brassTime.integer <= 0 ) {
return;
}
for ( i = 0; i < isHC + 1; i++ ) {
float waterScale = 1.0f;
le = CG_AllocLocalEntity();
re = &le->refEntity;
velocity[0] = 60 + 60 * crandom();
if ( i == 0 ) {
velocity[1] = 40 + 10 * crandom();
} else {
velocity[1] = -40 + 10 * crandom();
}
velocity[2] = 100 + 50 * crandom();
le->leType = LE_FRAGMENT;
le->startTime = cg.time;
le->endTime = le->startTime + cg_brassTime.integer*3 + cg_brassTime.integer * random();
le->pos.trType = TR_GRAVITY;
le->pos.trTime = cg.time;
AnglesToAxis( cent->lerpAngles, v );
offset[0] = 8;
offset[1] = 0;
offset[2] = 24;
// NiceAss: Added for better starting location of brass
if (cg.predictedPlayerState.pm_flags & PMF_DUCKED) {
offset[2] -= 14;
}
2001-05-06 20:50:27 +00:00
xoffset[0] = offset[0] * v[0][0] + offset[1] * v[1][0] + offset[2] * v[2][0];
xoffset[1] = offset[0] * v[0][1] + offset[1] * v[1][1] + offset[2] * v[2][1];
xoffset[2] = offset[0] * v[0][2] + offset[1] * v[1][2] + offset[2] * v[2][2];
VectorAdd( cent->lerpOrigin, xoffset, re->origin );
VectorCopy( re->origin, le->pos.trBase );
if ( CG_PointContents( re->origin, -1 ) & CONTENTS_WATER ) {
waterScale = 0.10f;
}
xvelocity[0] = velocity[0] * v[0][0] + velocity[1] * v[1][0] + velocity[2] * v[2][0];
xvelocity[1] = velocity[0] * v[0][1] + velocity[1] * v[1][1] + velocity[2] * v[2][1];
xvelocity[2] = velocity[0] * v[0][2] + velocity[1] * v[1][2] + velocity[2] * v[2][2];
VectorScale( xvelocity, waterScale, le->pos.trDelta );
AxisCopy( axisDefault, re->axis );
re->hModel = cgs.media.shotgunBrassModel;
le->bounceFactor = 0.3f;
le->angles.trType = TR_LINEAR;
le->angles.trTime = cg.time;
le->angles.trBase[0] = rand()&31;
le->angles.trBase[1] = rand()&31;
le->angles.trBase[2] = rand()&31;
le->angles.trDelta[0] = 1;
le->angles.trDelta[1] = 0.5;
le->angles.trDelta[2] = 0;
le->leFlags = LEF_TUMBLE;
le->leBounceSoundType = LEBS_BRASS;
le->leMarkType = LEMT_NONE;
}
}
#ifdef MISSIONPACK
/*
==========================
CG_NailgunEjectBrass
==========================
*/
static void CG_NailgunEjectBrass( centity_t *cent ) {
localEntity_t *smoke;
vec3_t origin;
vec3_t v[3];
vec3_t offset;
vec3_t xoffset;
vec3_t up;
AnglesToAxis( cent->lerpAngles, v );
offset[0] = 0;
offset[1] = -12;
offset[2] = 24;
xoffset[0] = offset[0] * v[0][0] + offset[1] * v[1][0] + offset[2] * v[2][0];
xoffset[1] = offset[0] * v[0][1] + offset[1] * v[1][1] + offset[2] * v[2][1];
xoffset[2] = offset[0] * v[0][2] + offset[1] * v[1][2] + offset[2] * v[2][2];
VectorAdd( cent->lerpOrigin, xoffset, origin );
VectorSet( up, 0, 0, 64 );
smoke = CG_SmokePuff( origin, up, 32, 1, 1, 1, 0.33f, 700, cg.time, 0, 0, cgs.media.smokePuffShader );
// use the optimized local entity add
smoke->leType = LE_SCALE_FADE;
}
#endif
/*
==========================
CG_RailTrail
==========================
*/
void CG_RailTrail( clientInfo_t *ci, vec3_t start, vec3_t end ) {
vec3_t axis[36], move, move2, next_move, vec, temp;
float len;
int i, j, skip;
2001-05-06 20:50:27 +00:00
localEntity_t *le;
refEntity_t *re;
#define RADIUS 4
#define ROTATION 1
#define SPACING 5
start[2] -= 4;
VectorCopy (start, move);
VectorSubtract (end, start, vec);
len = VectorNormalize (vec);
PerpendicularVector(temp, vec);
for (i = 0 ; i < 36; i++) {
RotatePointAroundVector(axis[i], vec, temp, i * 10);//banshee 2.4 was 10
}
2001-05-06 20:50:27 +00:00
le = CG_AllocLocalEntity();
re = &le->refEntity;
le->leType = LE_FADE_RGB;
le->startTime = cg.time;
le->endTime = cg.time + cg_railTrailTime.value;
le->lifeRate = 1.0 / ( le->endTime - le->startTime );
re->shaderTime = cg.time / 1000.0f;
re->reType = RT_RAIL_CORE;
re->customShader = cgs.media.railCoreShader;
2001-05-06 20:50:27 +00:00
VectorCopy( start, re->origin );
VectorCopy( end, re->oldorigin );
re->shaderRGBA[0] = ci->color1[0] * 255;
re->shaderRGBA[1] = ci->color1[1] * 255;
re->shaderRGBA[2] = ci->color1[2] * 255;
re->shaderRGBA[3] = 255;
2001-05-06 20:50:27 +00:00
le->color[0] = ci->color1[0] * 0.75;
le->color[1] = ci->color1[1] * 0.75;
le->color[2] = ci->color1[2] * 0.75;
2001-05-06 20:50:27 +00:00
le->color[3] = 1.0f;
AxisClear( re->axis );
VectorMA(move, 20, vec, move);
VectorCopy(move, next_move);
VectorScale (vec, SPACING, vec);
if (cg_oldRail.integer != 0) {
// nudge down a bit so it isn't exactly in center
re->origin[2] -= 8;
re->oldorigin[2] -= 8;
return;
}
skip = -1;
j = 18;
for (i = 0; i < len; i += SPACING) {
if (i != skip) {
skip = i + SPACING;
2001-05-06 20:50:27 +00:00
le = CG_AllocLocalEntity();
re = &le->refEntity;
le->leFlags = LEF_PUFF_DONT_SCALE;
le->leType = LE_MOVE_SCALE_FADE;
2001-05-06 20:50:27 +00:00
le->startTime = cg.time;
le->endTime = cg.time + (i>>1) + 600;
2001-05-06 20:50:27 +00:00
le->lifeRate = 1.0 / ( le->endTime - le->startTime );
re->shaderTime = cg.time / 1000.0f;
re->reType = RT_SPRITE;
re->radius = 1.1f;
re->customShader = cgs.media.railRingsShader;
2001-05-06 20:50:27 +00:00
re->shaderRGBA[0] = ci->color2[0] * 255;
re->shaderRGBA[1] = ci->color2[1] * 255;
re->shaderRGBA[2] = ci->color2[2] * 255;
re->shaderRGBA[3] = 255;
2001-05-06 20:50:27 +00:00
le->color[0] = ci->color2[0] * 0.75;
le->color[1] = ci->color2[1] * 0.75;
le->color[2] = ci->color2[2] * 0.75;
le->color[3] = 1.0f;
2001-05-06 20:50:27 +00:00
le->pos.trType = TR_LINEAR;
le->pos.trTime = cg.time;
2001-05-06 20:50:27 +00:00
VectorCopy( move, move2);
VectorMA(move2, RADIUS , axis[j], move2);
VectorCopy(move2, le->pos.trBase);
le->pos.trDelta[0] = axis[j][0]*6;
le->pos.trDelta[1] = axis[j][1]*6;
le->pos.trDelta[2] = axis[j][2]*6;
}
VectorAdd (move, vec, move);
j = j + ROTATION < 36 ? j + ROTATION : (j + ROTATION) % 36;
}
2001-05-06 20:50:27 +00:00
}
/*
==========================
CG_RocketTrail
==========================
*/
static void CG_RocketTrail( centity_t *ent, const weaponInfo_t *wi ) {
int step;
vec3_t origin, lastPos;
int t;
int startTime, contents;
int lastContents;
entityState_t *es;
vec3_t up;
localEntity_t *smoke;
if ( cg_noProjectileTrail.integer ) {
return;
}
2001-05-06 20:50:27 +00:00
up[0] = 0;
up[1] = 0;
up[2] = 0;
step = 50;
es = &ent->currentState;
startTime = ent->trailTime;
t = step * ( (startTime + step) / step );
CG_EvaluateTrajectory( &es->pos, cg.time, origin );
2001-05-06 20:50:27 +00:00
contents = CG_PointContents( origin, -1 );
// if object (e.g. grenade) is stationary, don't toss up smoke
if ( es->pos.trType == TR_STATIONARY ) {
ent->trailTime = cg.time;
return;
}
CG_EvaluateTrajectory( &es->pos, ent->trailTime, lastPos );
2001-05-06 20:50:27 +00:00
lastContents = CG_PointContents( lastPos, -1 );
ent->trailTime = cg.time;
if ( contents & ( CONTENTS_WATER | CONTENTS_SLIME | CONTENTS_LAVA ) ) {
if ( contents & lastContents & CONTENTS_WATER ) {
CG_BubbleTrail( lastPos, origin, 8 );
}
return;
}
for ( ; t <= ent->trailTime ; t += step ) {
CG_EvaluateTrajectory( &es->pos, t, lastPos );
2001-05-06 20:50:27 +00:00
smoke = CG_SmokePuff( lastPos, up,
wi->trailRadius,
2001-05-06 20:50:27 +00:00
1, 1, 1, 0.33f,
wi->wiTrailTime,
2001-05-06 20:50:27 +00:00
t,
0,
0,
2001-05-06 20:50:27 +00:00
cgs.media.smokePuffShader );
// use the optimized local entity add
smoke->leType = LE_SCALE_FADE;
}
}
#ifdef MISSIONPACK
/*
==========================
CG_NailTrail
==========================
*/
static void CG_NailTrail( centity_t *ent, const weaponInfo_t *wi ) {
int step;
vec3_t origin, lastPos;
int t;
int startTime, contents;
int lastContents;
entityState_t *es;
vec3_t up;
localEntity_t *smoke;
if ( cg_noProjectileTrail.integer ) {
return;
}
2001-05-06 20:50:27 +00:00
up[0] = 0;
up[1] = 0;
up[2] = 0;
step = 50;
es = &ent->currentState;
startTime = ent->trailTime;
t = step * ( (startTime + step) / step );
CG_EvaluateTrajectory( &es->pos, cg.time, origin );
2001-05-06 20:50:27 +00:00
contents = CG_PointContents( origin, -1 );
// if object (e.g. grenade) is stationary, don't toss up smoke
if ( es->pos.trType == TR_STATIONARY ) {
ent->trailTime = cg.time;
return;
}
CG_EvaluateTrajectory( &es->pos, ent->trailTime, lastPos );
2001-05-06 20:50:27 +00:00
lastContents = CG_PointContents( lastPos, -1 );
ent->trailTime = cg.time;
if ( contents & ( CONTENTS_WATER | CONTENTS_SLIME | CONTENTS_LAVA ) ) {
if ( contents & lastContents & CONTENTS_WATER ) {
CG_BubbleTrail( lastPos, origin, 8 );
}
return;
}
for ( ; t <= ent->trailTime ; t += step ) {
CG_EvaluateTrajectory( &es->pos, t, lastPos );
2001-05-06 20:50:27 +00:00
smoke = CG_SmokePuff( lastPos, up,
wi->trailRadius,
2001-05-06 20:50:27 +00:00
1, 1, 1, 0.33f,
wi->wiTrailTime,
2001-05-06 20:50:27 +00:00
t,
0,
0,
2001-05-06 20:50:27 +00:00
cgs.media.nailPuffShader );
// use the optimized local entity add
smoke->leType = LE_SCALE_FADE;
}
}
#endif
/*
==========================
CG_NailTrail
==========================
*/
static void CG_PlasmaTrail( centity_t *cent, const weaponInfo_t *wi ) {
localEntity_t *le;
refEntity_t *re;
entityState_t *es;
vec3_t velocity, xvelocity, origin;
vec3_t offset, xoffset;
vec3_t v[3];
int t, startTime, step;
float waterScale = 1.0f;
if ( cg_noProjectileTrail.integer || cg_oldPlasma.integer ) {
return;
}
step = 50;
es = &cent->currentState;
startTime = cent->trailTime;
t = step * ( (startTime + step) / step );
CG_EvaluateTrajectory( &es->pos, cg.time, origin );
le = CG_AllocLocalEntity();
re = &le->refEntity;
velocity[0] = 60 - 120 * crandom();
velocity[1] = 40 - 80 * crandom();
velocity[2] = 100 - 200 * crandom();
le->leType = LE_MOVE_SCALE_FADE;
le->leFlags = LEF_TUMBLE;
le->leBounceSoundType = LEBS_NONE;
le->leMarkType = LEMT_NONE;
le->startTime = cg.time;
le->endTime = le->startTime + 600;
le->pos.trType = TR_GRAVITY;
le->pos.trTime = cg.time;
AnglesToAxis( cent->lerpAngles, v );
offset[0] = 2;
offset[1] = 2;
offset[2] = 2;
xoffset[0] = offset[0] * v[0][0] + offset[1] * v[1][0] + offset[2] * v[2][0];
xoffset[1] = offset[0] * v[0][1] + offset[1] * v[1][1] + offset[2] * v[2][1];
xoffset[2] = offset[0] * v[0][2] + offset[1] * v[1][2] + offset[2] * v[2][2];
VectorAdd( origin, xoffset, re->origin );
VectorCopy( re->origin, le->pos.trBase );
if ( CG_PointContents( re->origin, -1 ) & CONTENTS_WATER ) {
waterScale = 0.10f;
}
xvelocity[0] = velocity[0] * v[0][0] + velocity[1] * v[1][0] + velocity[2] * v[2][0];
xvelocity[1] = velocity[0] * v[0][1] + velocity[1] * v[1][1] + velocity[2] * v[2][1];
xvelocity[2] = velocity[0] * v[0][2] + velocity[1] * v[1][2] + velocity[2] * v[2][2];
VectorScale( xvelocity, waterScale, le->pos.trDelta );
AxisCopy( axisDefault, re->axis );
re->shaderTime = cg.time / 1000.0f;
re->reType = RT_SPRITE;
re->radius = 0.25f;
re->customShader = cgs.media.railRingsShader;
le->bounceFactor = 0.3f;
re->shaderRGBA[0] = wi->flashDlightColor[0] * 63;
re->shaderRGBA[1] = wi->flashDlightColor[1] * 63;
re->shaderRGBA[2] = wi->flashDlightColor[2] * 63;
re->shaderRGBA[3] = 63;
le->color[0] = wi->flashDlightColor[0] * 0.2;
le->color[1] = wi->flashDlightColor[1] * 0.2;
le->color[2] = wi->flashDlightColor[2] * 0.2;
le->color[3] = 0.25f;
le->angles.trType = TR_LINEAR;
le->angles.trTime = cg.time;
le->angles.trBase[0] = rand()&31;
le->angles.trBase[1] = rand()&31;
le->angles.trBase[2] = rand()&31;
le->angles.trDelta[0] = 1;
le->angles.trDelta[1] = 0.5;
le->angles.trDelta[2] = 0;
}
2001-05-06 20:50:27 +00:00
/*
==========================
CG_GrappleTrail
==========================
*/
void CG_GrappleTrail( centity_t *ent, const weaponInfo_t *wi ) {
vec3_t origin;
entityState_t *es;
vec3_t forward, up;
refEntity_t beam;
es = &ent->currentState;
CG_EvaluateTrajectory( &es->pos, cg.time, origin );
2001-05-06 20:50:27 +00:00
ent->trailTime = cg.time;
memset( &beam, 0, sizeof( beam ) );
//FIXME adjust for muzzle position
VectorCopy ( cg_entities[ ent->currentState.otherEntityNum ].lerpOrigin, beam.origin );
beam.origin[2] += 26;
AngleVectors( cg_entities[ ent->currentState.otherEntityNum ].lerpAngles, forward, NULL, up );
VectorMA( beam.origin, -6, up, beam.origin );
VectorCopy( origin, beam.oldorigin );
if (Distance( beam.origin, beam.oldorigin ) < 64 )
return; // Don't draw if close
beam.reType = RT_LIGHTNING;
beam.customShader = cgs.media.lightningShader;
AxisClear( beam.axis );
beam.shaderRGBA[0] = 0xff;
beam.shaderRGBA[1] = 0xff;
beam.shaderRGBA[2] = 0xff;
beam.shaderRGBA[3] = 0xff;
trap_R_AddRefEntityToScene( &beam );
}
/*
==========================
CG_GrenadeTrail
==========================
*/
static void CG_GrenadeTrail( centity_t *ent, const weaponInfo_t *wi ) {
CG_RocketTrail( ent, wi );
2001-05-06 20:50:27 +00:00
}
/*
=================
CG_RegisterWeapon
The server says this item is used on this level
=================
*/
void CG_RegisterWeapon( int weaponNum ) {
weaponInfo_t *weaponInfo;
gitem_t *item, *ammo;
char path[MAX_QPATH];
vec3_t mins, maxs;
2001-07-28 02:49:18 +00:00
// QUARANTINE - Weapon Animations - Added Variable
char filename[MAX_QPATH]; //Used to open animation.cfg files
// END
2001-05-06 20:50:27 +00:00
int i;
qboolean weapAnimLoad = qtrue;
2001-05-06 20:50:27 +00:00
weaponInfo = &cg_weapons[weaponNum];
if ( weaponNum == 0 ) {
return;
}
if ( weaponInfo->registered ) {
return;
}
memset( weaponInfo, 0, sizeof( *weaponInfo ) );
weaponInfo->registered = qtrue;
2001-05-06 20:50:27 +00:00
for ( item = bg_itemlist + 1 ; item->classname ; item++ ) {
if ( item->giType == IT_WEAPON && item->giTag == weaponNum ) {
weaponInfo->item = item;
break;
}
}
if ( !item->classname ) {
CG_Error( "Couldn't find weapon %i", weaponNum );
}
CG_RegisterItemVisuals( item - bg_itemlist );
// load cmodel before model so filecache works
weaponInfo->weaponModel = trap_R_RegisterModel( item->world_model[0] );
// calc midpoint for rotation
trap_R_ModelBounds( weaponInfo->weaponModel, mins, maxs );
for ( i = 0 ; i < 3 ; i++ ) {
weaponInfo->weaponMidpoint[i] = mins[i] + 0.5 * ( maxs[i] - mins[i] );
}
weaponInfo->weaponIcon = trap_R_RegisterShader( item->icon );
weaponInfo->ammoIcon = trap_R_RegisterShader( item->icon );
for ( ammo = bg_itemlist + 1 ; ammo->classname ; ammo++ ) {
if ( ammo->giType == IT_AMMO && ammo->giTag == weaponNum ) {
break;
}
//Elder: hack for handcannon to use M3 ammo icon
else if (weaponNum == WP_HANDCANNON && ammo->giType == IT_AMMO && ammo->giTag == WP_M3)
{
break;
}
//Elder: hack for akimbos to use MK23 ammo icon
else if (weaponNum == WP_AKIMBO && ammo->giType == IT_AMMO && ammo->giTag == WP_PISTOL)
{
break;
}
2001-05-06 20:50:27 +00:00
}
if ( ammo->classname && ammo->world_model[0] ) {
weaponInfo->ammoModel = trap_R_RegisterModel( ammo->world_model[0] );
//Elder: added
weaponInfo->ammoIcon = trap_R_RegisterShader( ammo->icon );
2001-05-06 20:50:27 +00:00
}
strcpy( path, item->world_model[0] );
COM_StripExtension( path, path );
strcat( path, "_flash.md3" );
weaponInfo->flashModel = trap_R_RegisterModel( path );
strcpy( path, item->world_model[0] );
COM_StripExtension( path, path );
strcat( path, "_barrel.md3" );
weaponInfo->barrelModel = trap_R_RegisterModel( path );
strcpy( path, item->world_model[0] );
COM_StripExtension( path, path );
strcat( path, "_hand.md3" );
weaponInfo->handsModel = trap_R_RegisterModel( path );
//Elder: added to cache 1st-person models
strcpy( path, item->world_model[0] );
COM_StripExtension( path, path );
strcat( path, "_1st.md3" );
weaponInfo->firstModel = trap_R_RegisterModel( path );
2001-05-06 20:50:27 +00:00
if ( !weaponInfo->handsModel ) {
weaponInfo->handsModel = trap_R_RegisterModel( "models/weapons2/shotgun/shotgun_hand.md3" );
}
//Elder: if no _1st model, point to the weaponModel... this may get funky :)
if ( !weaponInfo->firstModel ) {
2001-08-18 20:01:21 +00:00
// Added warning message
CG_Printf(" ^3Warning: %s first-person model not found; using world model^7\n", weaponInfo->item->pickup_name);
weaponInfo->firstModel = weaponInfo->weaponModel;
2001-05-06 20:50:27 +00:00
}
weaponInfo->loopFireSound = qfalse;
2001-05-06 20:50:27 +00:00
switch ( weaponNum ) {
2001-11-11 07:23:16 +00:00
//Blaze: Reaction Pistol
2001-05-06 20:50:27 +00:00
case WP_PISTOL:
MAKERGB( weaponInfo->flashDlightColor, 1, 1, 0 );
weaponInfo->flashSound[0] = trap_S_RegisterSound( "sound/weapons/mk23/mk23fire.wav", qfalse );
2001-11-11 07:23:16 +00:00
weaponInfo->worldReloadSound[0] = trap_S_RegisterSound( "sound/weapons/mk23/mk23reload.wav", qfalse );
2001-05-06 20:50:27 +00:00
weaponInfo->ejectBrassFunc = CG_MachineGunEjectBrass;
cgs.media.bulletExplosionShader = trap_R_RegisterShader( "bulletExplosion" );
// Load the animation information
2001-07-28 02:49:18 +00:00
Com_sprintf( filename, sizeof(filename), "models/weapons2/mk23/animation.cfg" );
if ( !CG_ParseWeaponAnimFile(filename, weaponInfo) ) {
Com_Printf("Failed to load weapon animation file %s\n", filename);
weapAnimLoad = qfalse;
}
// Load sound information -- ALWAYS DO THIS AFTER THE ANIMATION
if (weapAnimLoad)
{
Com_sprintf( filename, sizeof(filename), "models/weapons2/mk23/sound.cfg" );
if ( !CG_ParseWeaponSoundFile(filename, weaponInfo) ) {
Com_Printf("Failed to load weapon sound file %s\n", filename);
}
}
else
{
Com_Printf("Could not load sound.cfg because animation.cfg loading failed\n");
2001-07-28 02:49:18 +00:00
}
2001-05-06 20:50:27 +00:00
break;
case WP_KNIFE:
MAKERGB( weaponInfo->flashDlightColor, 1, 0.70f, 0 );
2001-05-06 20:50:27 +00:00
weaponInfo->flashSound[0] = trap_S_RegisterSound( "sound/weapons/knife/slash.wav", qfalse );
weaponInfo->missileModel = trap_R_RegisterModel("models/weapons2/knife/knife.md3");
// Load the animation information
Com_sprintf( filename, sizeof(filename), "models/weapons2/knife/animation.cfg" );
if ( !CG_ParseWeaponAnimFile(filename, weaponInfo) ) {
Com_Printf("Failed to load weapon animation file %s\n", filename);
weapAnimLoad = qfalse;
}
2001-05-06 20:50:27 +00:00
//weaponInfo->missileTrailFunc = CG_GrenadeTrail;
//cgs.media.grenadeExplosionShader = trap_R_RegisterShader( "grenadeExplosion" );
//weaponInfo->wiTrailTime = 700;
//weaponInfo->trailRadius = 32;
2001-05-06 20:50:27 +00:00
break;
2001-05-06 20:50:27 +00:00
case WP_M4:
//Elder: added
MAKERGB( weaponInfo->flashDlightColor, 1, 1, 0 );
2001-05-06 20:50:27 +00:00
weaponInfo->flashSound[0] = trap_S_RegisterSound( "sound/weapons/m4/m4fire.wav", qfalse );
2001-11-11 07:23:16 +00:00
weaponInfo->worldReloadSound[0] = trap_S_RegisterSound( "sound/weapons/m4/m4reload.wav", qfalse );
2001-05-06 20:50:27 +00:00
weaponInfo->ejectBrassFunc = CG_MachineGunEjectBrass;
cgs.media.bulletExplosionShader = trap_R_RegisterShader( "bulletExplosion" );
Com_sprintf( filename, sizeof(filename), "models/weapons2/m4/animation.cfg" );
if ( !CG_ParseWeaponAnimFile(filename, weaponInfo) ) {
Com_Printf("Failed to load weapon animation file %s\n", filename);
weapAnimLoad = qfalse;
}
if (weapAnimLoad) {
Com_sprintf( filename, sizeof(filename), "models/weapons2/m4/sound.cfg" );
if ( !CG_ParseWeaponSoundFile(filename, weaponInfo) ) {
Com_Printf("Failed to load weapon sound file %s\n", filename);
}
} else {
Com_Printf("Could not load sound.cfg because animation.cfg loading failed\n");
}
2001-05-06 20:50:27 +00:00
break;
2001-05-06 20:50:27 +00:00
case WP_SSG3000:
//Elder: added
MAKERGB( weaponInfo->flashDlightColor, 1, 0.5f, 0 );
2001-05-06 20:50:27 +00:00
weaponInfo->flashSound[0] = trap_S_RegisterSound( "sound/weapons/ssg3000/ssgfire.wav", qfalse );
2001-11-11 07:23:16 +00:00
weaponInfo->worldReloadSound[0] = trap_S_RegisterSound( "sound/weapons/ssg3000/ssgbolt.wav", qfalse );
weaponInfo->worldReloadSound[1] = trap_S_RegisterSound( "sound/weapons/ssg3000/ssgin.wav", qfalse );
weaponInfo->worldReloadSound[2] = trap_S_RegisterSound( "sound/weapons/ssg3000/ssgbolt.wav", qfalse );
2001-05-06 20:50:27 +00:00
weaponInfo->ejectBrassFunc = CG_MachineGunEjectBrass;
cgs.media.bulletExplosionShader = trap_R_RegisterShader( "bulletExplosion" );
2001-08-25 00:05:20 +00:00
Com_sprintf( filename, sizeof(filename), "models/weapons2/ssg3000/animation.cfg" );
if ( !CG_ParseWeaponAnimFile(filename, weaponInfo) ) {
Com_Printf("Failed to load weapon animation file %s\n", filename);
}
// Load sound information -- ALWAYS DO THIS AFTER THE ANIMATION
if (weapAnimLoad) {
Com_sprintf( filename, sizeof(filename), "models/weapons2/ssg3000/sound.cfg" );
if ( !CG_ParseWeaponSoundFile(filename, weaponInfo) ) {
Com_Printf("Failed to load weapon sound file %s\n", filename);
}
} else {
Com_Printf("Could not load sound.cfg because animation.cfg loading failed\n");
}
2001-05-06 20:50:27 +00:00
break;
2001-05-06 20:50:27 +00:00
case WP_MP5:
//Elder: added
MAKERGB( weaponInfo->flashDlightColor, 1, 0.75f, 0 );
weaponInfo->flashSound[0] = trap_S_RegisterSound( "sound/weapons/mp5/mp5fire.wav", qfalse );
2001-11-11 07:23:16 +00:00
weaponInfo->worldReloadSound[0] = trap_S_RegisterSound( "sound/weapons/mp5/mp5reload.wav", qfalse );
2001-05-06 20:50:27 +00:00
weaponInfo->ejectBrassFunc = CG_MachineGunEjectBrass;
cgs.media.bulletExplosionShader = trap_R_RegisterShader( "bulletExplosion" );
2001-10-30 23:23:04 +00:00
Com_sprintf( filename, sizeof(filename), "models/weapons2/mp5/animation.cfg" );
if ( !CG_ParseWeaponAnimFile(filename, weaponInfo) ) {
Com_Printf("Failed to load weapon animation file %s\n", filename);
weapAnimLoad = qfalse;
}
2001-11-08 02:37:11 +00:00
if (weapAnimLoad) {
Com_sprintf( filename, sizeof(filename), "models/weapons2/mp5/sound.cfg" );
if ( !CG_ParseWeaponSoundFile(filename, weaponInfo) ) {
Com_Printf("Failed to load weapon sound file %s\n", filename);
}
} else {
Com_Printf("Could not load sound.cfg because animation.cfg loading failed\n");
}
2001-05-06 20:50:27 +00:00
break;
2001-05-06 20:50:27 +00:00
case WP_HANDCANNON:
//Elder: added
MAKERGB( weaponInfo->flashDlightColor, 1, 1, 0 );
//Elder: changed to hcfire from cannon_fire
weaponInfo->flashSound[0] = trap_S_RegisterSound( "sound/weapons/handcannon/hcfire.wav", qfalse );
2001-11-11 07:23:16 +00:00
weaponInfo->worldReloadSound[0] = trap_S_RegisterSound( "sound/weapons/handcannon/hcreload.wav", qfalse );
//weaponInfo->ejectBrassFunc = CG_ShotgunEjectBrass;
2001-05-06 20:50:27 +00:00
cgs.media.bulletExplosionShader = trap_R_RegisterShader( "bulletExplosion" );
2001-08-17 14:28:52 +00:00
Com_sprintf( filename, sizeof(filename), "models/weapons2/handcannon/animation.cfg" );
if ( !CG_ParseWeaponAnimFile(filename, weaponInfo) ) {
Com_Printf("Failed to load weapon animation file %s\n", filename);
weapAnimLoad = qfalse;
2001-08-17 14:28:52 +00:00
}
if (weapAnimLoad) {
Com_sprintf( filename, sizeof(filename), "models/weapons2/handcannon/sound.cfg" );
if ( !CG_ParseWeaponSoundFile(filename, weaponInfo) ) {
Com_Printf("Failed to load weapon sound file %s\n", filename);
}
} else {
Com_Printf("Could not load sound.cfg because animation.cfg loading failed\n");
}
2001-05-06 20:50:27 +00:00
break;
2001-05-06 20:50:27 +00:00
case WP_M3:
MAKERGB( weaponInfo->flashDlightColor, 1, 1, 0 );
weaponInfo->flashSound[0] = trap_S_RegisterSound( "sound/weapons/m3/m3fire.wav", qfalse );
2001-11-11 07:23:16 +00:00
weaponInfo->worldReloadSound[1] = trap_S_RegisterSound( "sound/weapons/m3/m3in.wav", qfalse );
2001-05-06 20:50:27 +00:00
weaponInfo->ejectBrassFunc = CG_ShotgunEjectBrass;
Com_sprintf( filename, sizeof(filename), "models/weapons2/m3/animation.cfg" );
if ( !CG_ParseWeaponAnimFile(filename, weaponInfo) ) {
Com_Printf("Failed to load weapon animation file %s\n", filename);
weapAnimLoad = qfalse;
}
if (weapAnimLoad) {
Com_sprintf( filename, sizeof(filename), "models/weapons2/m3/sound.cfg" );
if ( !CG_ParseWeaponSoundFile(filename, weaponInfo) ) {
Com_Printf("Failed to load weapon sound file %s\n", filename);
}
} else {
Com_Printf("Could not load sound.cfg because animation.cfg loading failed\n");
}
2001-05-06 20:50:27 +00:00
break;
2001-05-06 20:50:27 +00:00
case WP_AKIMBO:
//Elder: added
MAKERGB( weaponInfo->flashDlightColor, 1, 1, 0.5f );
// Elder: no more pseudo-dual sound needed :)
weaponInfo->flashSound[0] = trap_S_RegisterSound( "sound/weapons/mk23/mk23fire.wav", qfalse );
2001-11-11 07:23:16 +00:00
weaponInfo->worldReloadSound[0] = trap_S_RegisterSound( "sound/weapons/akimbo/akimboreload.wav", qfalse );
2001-05-06 20:50:27 +00:00
weaponInfo->ejectBrassFunc = CG_MachineGunEjectBrass;
cgs.media.bulletExplosionShader = trap_R_RegisterShader( "bulletExplosion" );
Com_sprintf( filename, sizeof(filename), "models/weapons2/akimbo/animation.cfg" );
if ( !CG_ParseWeaponAnimFile(filename, weaponInfo) ) {
Com_Printf("Failed to load weapon animation file %s\n", filename);
weapAnimLoad = qfalse;
}
// Load sound information -- ALWAYS DO THIS AFTER THE ANIMATION
if (weapAnimLoad) {
Com_sprintf( filename, sizeof(filename), "models/weapons2/akimbo/sound.cfg" );
if ( !CG_ParseWeaponSoundFile(filename, weaponInfo) ) {
Com_Printf("Failed to load weapon sound file %s\n", filename);
}
} else {
Com_Printf("Could not load sound.cfg because animation.cfg loading failed\n");
}
2001-05-06 20:50:27 +00:00
break;
2001-05-06 20:50:27 +00:00
case WP_GRENADE:
//Use the projectile model
weaponInfo->missileModel = trap_R_RegisterModel( "models/weapons2/grenade/gren_projectile.md3" );
2001-11-11 07:23:16 +00:00
//Elder: removed smoke for the last time! :)
//weaponInfo->missileTrailFunc = CG_GrenadeTrail;
2001-05-06 20:50:27 +00:00
weaponInfo->wiTrailTime = 700;
weaponInfo->trailRadius = 32;
MAKERGB( weaponInfo->flashDlightColor, 1, 0.70f, 0 );
//weaponInfo->flashSound[0] = trap_S_RegisterSound( "sound/weapons/grenade/gren_throw.wav", qfalse );
2001-05-06 20:50:27 +00:00
cgs.media.grenadeExplosionShader = trap_R_RegisterShader( "grenadeExplosion" );
// Load the animation information
Com_sprintf( filename, sizeof(filename), "models/weapons2/grenade/animation.cfg" );
if ( !CG_ParseWeaponAnimFile(filename, weaponInfo) ) {
Com_Printf("Failed to load weapon animation file %s\n", filename);
weapAnimLoad = qfalse;
}
// Load sound information -- ALWAYS DO THIS AFTER THE ANIMATION
if (weapAnimLoad) {
Com_sprintf( filename, sizeof(filename), "models/weapons2/grenade/sound.cfg" );
if ( !CG_ParseWeaponSoundFile(filename, weaponInfo) ) {
Com_Printf("Failed to load weapon sound file %s\n", filename);
}
} else {
Com_Printf("Could not load sound.cfg because animation.cfg loading failed\n");
}
2001-05-06 20:50:27 +00:00
break;
default:
//MAKERGB( weaponInfo->flashDlightColor, 1, 1, 1 );
//weaponInfo->flashSound[0] = trap_S_RegisterSound( "sound/weapons/rocket/rocklf1a.wav", qfalse );
break;
}
}
/*
=================
CG_RegisterItemVisuals
The server says this item is used on this level
=================
*/
void CG_RegisterItemVisuals( int itemNum ) {
itemInfo_t *itemInfo;
gitem_t *item;
itemInfo = &cg_items[ itemNum ];
if ( itemInfo->registered ) {
return;
}
item = &bg_itemlist[ itemNum ];
memset( itemInfo, 0, sizeof( &itemInfo ) );
itemInfo->registered = qtrue;
itemInfo->models[0] = trap_R_RegisterModel( item->world_model[0] );
itemInfo->icon = trap_R_RegisterShader( item->icon );
if ( item->giType == IT_WEAPON ) {
CG_RegisterWeapon( item->giTag );
}
//
// powerups have an accompanying ring or sphere
//
if ( item->giType == IT_POWERUP || item->giType == IT_HEALTH ||
2001-05-06 20:50:27 +00:00
item->giType == IT_ARMOR || item->giType == IT_HOLDABLE ) {
if ( item->world_model[1] ) {
itemInfo->models[1] = trap_R_RegisterModel( item->world_model[1] );
}
}
}
/*
========================================================================================
VIEW WEAPON
========================================================================================
*/
/*
=================
CG_MapTorsoToWeaponFrame
=================
*/
static int CG_MapTorsoToWeaponFrame( clientInfo_t *ci, int frame ) {
// change weapon
if ( frame >= ci->animations[TORSO_DROP].firstFrame
&& frame < ci->animations[TORSO_DROP].firstFrame + 9 ) {
return frame - ci->animations[TORSO_DROP].firstFrame + 6;
}
2001-05-06 20:50:27 +00:00
// stand attack
if ( frame >= ci->animations[TORSO_ATTACK].firstFrame
&& frame < ci->animations[TORSO_ATTACK].firstFrame + 6 ) {
return 1 + frame - ci->animations[TORSO_ATTACK].firstFrame;
}
2001-05-06 20:50:27 +00:00
// stand attack 2
if ( frame >= ci->animations[TORSO_ATTACK2].firstFrame
&& frame < ci->animations[TORSO_ATTACK2].firstFrame + 6 ) {
return 1 + frame - ci->animations[TORSO_ATTACK2].firstFrame;
2001-05-06 20:50:27 +00:00
}
2001-05-06 20:50:27 +00:00
return 0;
}
/*
==============
CG_CalculateWeaponPosition
==============
*/
static void CG_CalculateWeaponPosition( vec3_t origin, vec3_t angles ) {
float scale;
int delta;
float fracsin;
VectorCopy( cg.refdef.vieworg, origin );
VectorCopy( cg.refdefViewAngles, angles );
// on odd legs, invert some angles
if ( cg.bobcycle & 1 ) {
scale = -cg.xyspeed;
} else {
scale = cg.xyspeed;
}
// gun angles from bobbing
angles[ROLL] += scale * cg.bobfracsin * 0.005;
angles[YAW] += scale * cg.bobfracsin * 0.01;
angles[PITCH] += cg.xyspeed * cg.bobfracsin * 0.005;
// drop the weapon when landing
delta = cg.time - cg.landTime;
if ( delta < LAND_DEFLECT_TIME ) {
origin[2] += cg.landChange*0.25 * delta / LAND_DEFLECT_TIME;
} else if ( delta < LAND_DEFLECT_TIME + LAND_RETURN_TIME ) {
origin[2] += cg.landChange*0.25 *
2001-05-06 20:50:27 +00:00
(LAND_DEFLECT_TIME + LAND_RETURN_TIME - delta) / LAND_RETURN_TIME;
}
#if 0
// drop the weapon when stair climbing
delta = cg.time - cg.stepTime;
if ( delta < STEP_TIME/2 ) {
origin[2] -= cg.stepChange*0.25 * delta / (STEP_TIME/2);
} else if ( delta < STEP_TIME ) {
origin[2] -= cg.stepChange*0.25 * (STEP_TIME - delta) / (STEP_TIME/2);
}
#endif
// idle drift
scale = cg.xyspeed + 40;
fracsin = sin( cg.time * 0.001 );
angles[ROLL] += scale * fracsin * 0.01;
angles[YAW] += scale * fracsin * 0.01;
angles[PITCH] += scale * fracsin * 0.01;
}
/*
===============
CG_LightningBolt
Origin will be the exact tag point, which is slightly
different than the muzzle point used for determining hits.
The cent should be the non-predicted cent if it is from the player,
so the endpoint will reflect the simulated strike (lagging the predicted
angle)
2001-12-17 15:06:41 +00:00
NiceAss; I've torn this function up with testing... ignore it =P
2001-05-06 20:50:27 +00:00
===============
*/
static void CG_LightningBolt( centity_t *cent, vec3_t origin ) {
/*
2001-05-06 20:50:27 +00:00
trace_t trace;
refEntity_t beam;
vec3_t forward;
vec3_t muzzlePoint, endPoint;
2001-12-17 15:06:41 +00:00
centity_t *client;
2001-05-06 20:50:27 +00:00
//Blaze: LG No longer exists
// if ( cent->currentState.weapon != WP_LIGHTNING ) {
// return;
// }
memset( &beam, 0, sizeof( beam ) );
2001-12-17 15:06:41 +00:00
//client = &cg_entities[cg.predictedPlayerState.clientNum];
//CG_Printf("%d \n", cg.predictedPlayerState.clientNum);
2001-05-06 20:50:27 +00:00
// find muzzle point for this frame
//CG_EvaluateTrajectory( &cent->currentState.apos, cg.time, cent->lerpAngles );
2001-05-06 20:50:27 +00:00
AngleVectors( cent->lerpAngles, forward, NULL, NULL );
// project forward by the lightning range
2001-12-17 15:06:41 +00:00
VectorMA( muzzlePoint, 8192, forward, endPoint );
2001-05-06 20:50:27 +00:00
// see if it hit a wall
CG_Trace( &trace, muzzlePoint, vec3_origin, vec3_origin, endPoint,
2001-05-06 20:50:27 +00:00
cent->currentState.number, MASK_SHOT );
// this is the endpoint
VectorCopy( trace.endpos, beam.oldorigin );
// use the provided origin, even though it may be slightly
// different than the muzzle origin
VectorCopy( origin, beam.origin );
beam.reType = RT_LIGHTNING;
beam.customShader = cgs.media.lightningShader;
trap_R_AddRefEntityToScene( &beam );
// add the impact flare if it hit something
if ( trace.fraction < 1.0 ) {
}
*/
2001-05-06 20:50:27 +00:00
}
/*
===============
CG_SpawnRailTrail
Origin will be the exact tag point, which is slightly
different than the muzzle point used for determining hits.
===============
*/
static void CG_SpawnRailTrail( centity_t *cent, vec3_t origin ) {
clientInfo_t *ci;
//Blaze: No more rail gun
//if ( cent->currentState.weapon != WP_RAILGUN ) {
// return;
//}
if ( !cent->pe.railgunFlash ) {
return;
}
cent->pe.railgunFlash = qtrue;
ci = &cgs.clientinfo[ cent->currentState.clientNum ];
//Blaze: No Rail Trail
//CG_RailTrail( ci, origin, cent->pe.railgunImpact );
}
/*
======================
CG_MachinegunSpinAngle
======================
*/
#define SPIN_SPEED 0.9
#define COAST_TIME 1000
static float CG_MachinegunSpinAngle( centity_t *cent ) {
int delta;
float angle;
float speed;
delta = cg.time - cent->pe.barrelTime;
if ( cent->pe.barrelSpinning ) {
angle = cent->pe.barrelAngle + delta * SPIN_SPEED;
} else {
if ( delta > COAST_TIME ) {
delta = COAST_TIME;
}
speed = 0.5 * ( SPIN_SPEED + (float)( COAST_TIME - delta ) / COAST_TIME );
angle = cent->pe.barrelAngle + delta * speed;
}
if ( cent->pe.barrelSpinning == !(cent->currentState.eFlags & EF_FIRING) ) {
cent->pe.barrelTime = cg.time;
cent->pe.barrelAngle = AngleMod( angle );
cent->pe.barrelSpinning = !!(cent->currentState.eFlags & EF_FIRING);
#ifdef MISSIONPACK
if ( cent->currentState.weapon == WP_CHAINGUN && !cent->pe.barrelSpinning ) {
trap_S_StartSound( NULL, cent->currentState.number, CHAN_WEAPON, trap_S_RegisterSound( "sound/weapons/vulcan/wvulwind.wav", qfalse ) );
}
#endif
}
return angle;
}
/*
========================
CG_AddWeaponWithPowerups
========================
*/
static void CG_AddWeaponWithPowerups( refEntity_t *gun, int powerups ) {
// add powerup effects
if ( powerups & ( 1 << PW_INVIS ) ) {
gun->customShader = cgs.media.invisShader;
trap_R_AddRefEntityToScene( gun );
} else {
trap_R_AddRefEntityToScene( gun );
if ( powerups & ( 1 << PW_BATTLESUIT ) ) {
gun->customShader = cgs.media.battleWeaponShader;
trap_R_AddRefEntityToScene( gun );
}
if ( powerups & ( 1 << PW_QUAD ) ) {
gun->customShader = cgs.media.quadWeaponShader;
trap_R_AddRefEntityToScene( gun );
}
}
}
/*
=============
CG_AddPlayerWeapon
Used for both the view weapon (ps is valid) and the world modelother character models (ps is NULL)
The main player will have this called for BOTH cases, so effects like light and
sound should only be done on the world model case.
=============
*/
void CG_AddPlayerWeapon( refEntity_t *parent, playerState_t *ps, centity_t *cent, int team ) {
refEntity_t gun;
//Blaze: Can remove this because no more spinning barrel
// refEntity_t barrel;
refEntity_t flash;
vec3_t angles;
// float frac;
2001-05-06 20:50:27 +00:00
weapon_t weaponNum;
weaponInfo_t *weapon;
centity_t *nonPredictedCent;
// int col;
weaponNum = cent->currentState.weapon;
CG_RegisterWeapon( weaponNum );
weapon = &cg_weapons[weaponNum];
// add the weapon
memset( &gun, 0, sizeof( gun ) );
VectorCopy( parent->lightingOrigin, gun.lightingOrigin );
gun.shadowPlane = parent->shadowPlane;
gun.renderfx = parent->renderfx;
// set custom shading for railgun refire rate
//if ( ps ) {
//Blaze: No Railgun
//if ( cg.predictedPlayerState.weapon == WP_RAILGUN
2001-05-06 20:50:27 +00:00
// && cg.predictedPlayerState.weaponstate == WEAPON_FIRING ) {
// float f;
// f = (float)cg.predictedPlayerState.weaponTime / 1500;
// gun.shaderRGBA[1] = 0;
// gun.shaderRGBA[0] =
2001-05-06 20:50:27 +00:00
// gun.shaderRGBA[2] = 255 * ( 1.0 - f );
//} else {
gun.shaderRGBA[0] = 255;
gun.shaderRGBA[1] = 255;
gun.shaderRGBA[2] = 255;
gun.shaderRGBA[3] = 255;
//}
//}
if (ps == NULL)
{
//Elder: We are in third person, use the third-person model (DEFAULT)
gun.hModel = weapon->weaponModel;
2001-05-06 20:50:27 +00:00
}
else {
//Elder: we are in first-person, use the first-person (NOT default) model
gun.hModel = weapon->firstModel;
2001-05-06 20:50:27 +00:00
}
2001-05-06 20:50:27 +00:00
if (!gun.hModel) {
return;
}
if ( !ps ) {
// add weapon ready sound
cent->pe.lightningFiring = qfalse;
if ( ( cent->currentState.eFlags & EF_FIRING ) && weapon->firingSound ) {
// lightning gun and guantlet make a different sound when fire is held down
trap_S_AddLoopingSound( cent->currentState.number, cent->lerpOrigin, vec3_origin, weapon->firingSound );
cent->pe.lightningFiring = qtrue;
} else if ( weapon->readySound ) {
trap_S_AddLoopingSound( cent->currentState.number, cent->lerpOrigin, vec3_origin, weapon->readySound );
}
}
2001-07-28 02:49:18 +00:00
//if (ps == NULL)
//{
2001-05-06 20:50:27 +00:00
/* resizing code
msec = cg.time - cent->miscTime;
if ( msec >= 0 && msec < ITEM_SCALEUP_TIME ) {
frac = (float)msec / ITEM_SCALEUP_TIME;
VectorScale( ent.axis[0], frac, ent.axis[0] );
VectorScale( ent.axis[1], frac, ent.axis[1] );
VectorScale( ent.axis[2], frac, ent.axis[2] );
ent.nonNormalizedAxes = qtrue;
} else {
frac = 1.0;
}
*/
//Elder: shouldn't have to do this anymore
/*
2001-05-06 20:50:27 +00:00
if (weaponNum == 4)//Blaze: Scale the Sniper Rifle down a bit
{
frac = 0.8f;
VectorScale(parent->axis[0], frac, parent->axis[0]);
//VectorScale(parent->axis[1], frac, parent->axis[1]);
//VectorScale(parent->axis[2], frac, parent->axis[2]);
}
*/
//}
2001-07-28 02:49:18 +00:00
if ( !ps ) {
CG_PositionEntityOnTag( &gun, parent, parent->hModel, "tag_weapon");
}
else {
2001-11-25 23:17:41 +00:00
2001-07-28 02:49:18 +00:00
CG_WeaponAnimation( cent, &gun.oldframe, &gun.frame, &gun.backlerp );
CG_PositionWeaponOnTag( &gun, parent, parent->hModel, "tag_weapon");
// Elder: Local sound events will sync perfectly here
// However, we must remember to ignore the ones pmove will generate
// (for other clients to hear) or we'll get some nasty echo
// Temp hack because it isn't fully implemented
if ( weapon->item->giTag == WP_PISTOL ||
weapon->item->giTag == WP_AKIMBO ||
weapon->item->giTag == WP_M4 ||
2001-11-08 02:37:11 +00:00
weapon->item->giTag == WP_MP5 ||
weapon->item->giTag == WP_M3 ||
weapon->item->giTag == WP_HANDCANNON ||
weapon->item->giTag == WP_SSG3000 ||
weapon->item->giTag == WP_GRENADE)
{
int i = 0;
qboolean noSound = qfalse;
2001-11-25 23:17:41 +00:00
while ( gun.frame != weapon->animationSounds->sfxInfo[i].frame )
{
if ( ++i == weapon->animationSounds->numFrames )
{
noSound = qtrue;
break;
}
}
// reset the current sound
if (cg.curSyncSound.played && cg.curSyncSound.frame != gun.frame)
cg.curSyncSound.played = qfalse;
if (!noSound)
{
// copy the struct to reset it
if (cg.curSyncSound.frame != gun.frame )
cg.curSyncSound = weapon->animationSounds->sfxInfo[i];
if (cg.curSyncSound.played == qfalse)
{
cg.curSyncSound.played = qtrue;
2001-11-25 23:17:41 +00:00
2001-11-11 07:23:16 +00:00
//CG_Printf("Playing a timed sound (%i %i %1.1f)\n", gun.frame, gun.oldframe, gun.backlerp);
trap_S_StartLocalSound ( cg.curSyncSound.sound, CHAN_WEAPON );
}
}
}
}
2001-05-06 20:50:27 +00:00
// Elder: break off here so we still have weapon animations on bolt out
// Elder: added ps so we see SSG in third-person zoomed
if (cg.zoomed && ps)
return;
// JBravo: Adding a fix from NiceAss (cg_drawgun affecting other player models)
if (!cg_drawGun.integer && ps) return;
2001-05-06 20:50:27 +00:00
CG_AddWeaponWithPowerups( &gun, cent->currentState.powerups );
// add the spinning barrel
//Blaze: No Spinning Barrel
/*
if ( weapon->barrelModel ) {
memset( &barrel, 0, sizeof( barrel ) );
VectorCopy( parent->lightingOrigin, barrel.lightingOrigin );
barrel.shadowPlane = parent->shadowPlane;
barrel.renderfx = parent->renderfx;
barrel.hModel = weapon->barrelModel;
angles[YAW] = 0;
angles[PITCH] = 0;
angles[ROLL] = CG_MachinegunSpinAngle( cent );
AnglesToAxis( angles, barrel.axis );
CG_PositionRotatedEntityOnTag( &barrel, &gun, weapon->weaponModel, "tag_barrel" );
CG_AddWeaponWithPowerups( &barrel, cent->currentState.powerups );
}*/
// make sure we aren't looking at cg.predictedPlayerEntity for LG
nonPredictedCent = &cg_entities[cent->currentState.clientNum];
// if the index of the nonPredictedCent is not the same as the clientNum
// then this is a fake player (like on teh single player podiums), so
// go ahead and use the cent
if( ( nonPredictedCent - cg_entities ) != cent->currentState.clientNum ) {
nonPredictedCent = cent;
}
// add the flash
//Blaze: no more of these
/*
if ( ( weaponNum == WP_LIGHTNING || weaponNum == WP_GAUNTLET || weaponNum == WP_GRAPPLING_HOOK )
&& ( nonPredictedCent->currentState.eFlags & EF_FIRING ) )
2001-05-06 20:50:27 +00:00
{
// continuous flash
} else {
// impulse flash
if ( cg.time - cent->muzzleFlashTime > MUZZLE_FLASH_TIME && !cent->pe.railgunFlash ) {
return;
}
}
*/
2001-12-17 15:06:41 +00:00
//CG_LightningBolt( nonPredictedCent, parent->lightingOrigin );
//Elder: re-added to fix loss of muzzle flashes!
// impulse flash
if ( cg.time - cent->muzzleFlashTime > MUZZLE_FLASH_TIME && !cent->pe.railgunFlash ) {
return;
}
2001-05-06 20:50:27 +00:00
memset( &flash, 0, sizeof( flash ) );
2001-05-06 20:50:27 +00:00
VectorCopy( parent->lightingOrigin, flash.lightingOrigin );
flash.shadowPlane = parent->shadowPlane;
flash.renderfx = parent->renderfx;
2001-05-06 20:50:27 +00:00
flash.hModel = weapon->flashModel;
if (!flash.hModel) {
return;
}
2001-05-06 20:50:27 +00:00
angles[YAW] = 0;
angles[PITCH] = 0;
angles[ROLL] = crandom() * 10;
AnglesToAxis( angles, flash.axis );
// colorize the railgun blast
//Blaze: No more railgun
/*
if ( weaponNum == WP_RAILGUN ) {
clientInfo_t *ci;
ci = &cgs.clientinfo[ cent->currentState.clientNum ];
flash.shaderRGBA[0] = 255 * ci->color1[0];
flash.shaderRGBA[1] = 255 * ci->color1[1];
flash.shaderRGBA[2] = 255 * ci->color1[2];
2001-05-06 20:50:27 +00:00
}
*/
//Blaze: No flash
//Elder: Yes flash - try this
//Elder: add conditional here so the dlight is still drawn when cg_RQ3_flash is 0
if ( cg_RQ3_flash.integer ) {
if (ps) {
// Elder: draw flash based on first-person view
if (ps->weapon == WP_AKIMBO)
{
// choose tag for akimbos
if (ps->stats[STAT_BURST])
CG_PositionRotatedEntityOnTag( &flash, &gun, weapon->firstModel, "tag_flash");
else
CG_PositionRotatedEntityOnTag( &flash, &gun, weapon->firstModel, "tag_flash2");
}
else
CG_PositionRotatedEntityOnTag( &flash, &gun, weapon->firstModel, "tag_flash");
// Make flash larger to compensate for depth hack
VectorScale( flash.axis[0], 2.0f, flash.axis[0] );
VectorScale( flash.axis[1], 2.0f, flash.axis[1] );
VectorScale( flash.axis[2], 2.0f, flash.axis[2] );
flash.nonNormalizedAxes = qtrue;
}
else {
//Elder: draw flash based on 3rd-person view
if ( weapon->item->giTag == WP_AKIMBO)
{
// choose tag for akimbos
if (cg.akimboFlash)
CG_PositionRotatedEntityOnTag( &flash, &gun, weapon->weaponModel, "tag_flash");
else
CG_PositionRotatedEntityOnTag( &flash, &gun, weapon->weaponModel, "tag_flash2");
}
else
CG_PositionRotatedEntityOnTag( &flash, &gun, weapon->weaponModel, "tag_flash");
}
trap_R_AddRefEntityToScene( &flash );
}
else {
//Elder: crappy way to get the dlight working w/o the flash model
if ( ps || cg.renderingThirdPerson ||
cent->currentState.number != cg.predictedPlayerState.clientNum ) {
if ( weapon->flashDlightColor[0] || weapon->flashDlightColor[1] || weapon->flashDlightColor[2] ) {
if (!cg_drawGun.value)trap_R_AddLightToScene( gun.origin, 300 + (rand()&31), weapon->flashDlightColor[0],
weapon->flashDlightColor[1], weapon->flashDlightColor[2] );
}
}
return;
}
2001-07-28 12:32:54 +00:00
if ( ps || cg.renderingThirdPerson || cent->currentState.number != cg.predictedPlayerState.clientNum ) {
2001-05-06 20:50:27 +00:00
// add lightning bolt
//Blaze: No need for this
// add rail trail
//Blaze: no need for this
//CG_SpawnRailTrail( cent, flash.origin );
if ( weapon->flashDlightColor[0] || weapon->flashDlightColor[1] || weapon->flashDlightColor[2] ) {
trap_R_AddLightToScene( flash.origin, 300 + (rand()&31), weapon->flashDlightColor[0],
weapon->flashDlightColor[1], weapon->flashDlightColor[2] );
}
}
}
/*
==============
CG_AddViewWeapon
Add the weapon, and flash for the player's view
==============
*/
void CG_AddViewWeapon( playerState_t *ps ) {
refEntity_t hand;
centity_t *cent;
//clientInfo_t *ci;
2001-05-06 20:50:27 +00:00
float fovOffset;
vec3_t angles;
weaponInfo_t *weapon;
2001-12-17 15:06:41 +00:00
// remove this:
//vec3_t origin;
2001-12-17 15:06:41 +00:00
2001-05-06 20:50:27 +00:00
//Blaze: Reaction vars for gun positions
//float rxn_gunx, rxn_guny, rxn_gunz;
2001-05-06 20:50:27 +00:00
if ( ps->persistant[PERS_TEAM] == TEAM_SPECTATOR ) {
return;
}
if ( ps->pm_type == PM_INTERMISSION ) {
return;
}
// no gun if in third person view or a camera is active
//if ( cg.renderingThirdPerson || cg.cameraMode) {
2001-05-06 20:50:27 +00:00
if ( cg.renderingThirdPerson ) {
return;
}
// allow the gun to be completely removed
// Elder: don't draw if throwing
// if ( !cg_drawGun.integer || (cg.snap->ps.stats[STAT_RQ3] & RQ3_THROWWEAPON) == RQ3_THROWWEAPON) {
//Blaze: removed cg_drawGun checks and moved into other code.
if ( (cg.snap->ps.stats[STAT_RQ3] & RQ3_THROWWEAPON) == RQ3_THROWWEAPON) {
2001-05-06 20:50:27 +00:00
//Blaze: Removed these
2001-12-17 15:06:41 +00:00
if ( cg.predictedPlayerState.eFlags & EF_FIRING ) {
2001-05-06 20:50:27 +00:00
// special hack for lightning gun...
2001-12-17 15:06:41 +00:00
//VectorCopy( cg.refdef.vieworg, origin );
//VectorMA( origin, -8, cg.refdef.viewaxis[2], origin );
//CG_LightningBolt( &cg_entities[ps->clientNum], origin );
}
2001-05-06 20:50:27 +00:00
return;
}
// Hawkins- don't draw gun if zoomed
// Elder: we'll break it off later because we still need to perform animation operations
/*
2001-05-06 20:50:27 +00:00
if(cg.zoomed)
return;
*/
2001-05-06 20:50:27 +00:00
// don't draw if testing a gun model
if ( cg.testGun ) {
return;
}
2001-11-13 05:32:07 +00:00
// Elder: temporarily removed because we never used it anyways
/*
//Blaze start: Reaction gun positioning
rxn_gunx = cg_gun_x.value;
2001-05-06 20:50:27 +00:00
rxn_guny = cg_gun_y.value;
rxn_gunz = cg_gun_z.value;
//Blaze: Gun xyz coords changes
switch(ps->weapon)
{
case WP_PISTOL:
2001-07-28 02:49:18 +00:00
rxn_gunx += 10;
2001-05-06 20:50:27 +00:00
break;
case WP_M4:
rxn_gunx += 10;
break;
case WP_SSG3000:
rxn_gunx += 10;
break;
case WP_AKIMBO:
rxn_gunx += 10;
}
2001-11-25 23:04:08 +00:00
if ((cg_RQ3_drawWeapon.integer > 1 ) && (cg_RQ3_drawWeapon.integer < 4 ))
2001-05-06 20:50:27 +00:00
{
rxn_guny = cg_gun_y.value + 4*(cg_RQ3_drawWeapon.integer-1);
2001-05-06 20:50:27 +00:00
}
2001-11-13 05:32:07 +00:00
*/
2001-05-06 20:50:27 +00:00
//Blaze end:
// drop gun lower at higher fov
if ( cg_fov.integer > 90 ) {
fovOffset = -0.2 * ( cg_fov.integer - 90 );
} else {
fovOffset = 0;
}
cent = &cg.predictedPlayerEntity; // &cg_entities[cg.snap->ps.clientNum];
CG_RegisterWeapon( ps->weapon );
weapon = &cg_weapons[ ps->weapon ];
memset (&hand, 0, sizeof(hand));
// set up gun position
CG_CalculateWeaponPosition( hand.origin, angles );
//Blaze start: reaction gun positioning
// VectorMA( hand.origin, rxn_gunx, cg.refdef.viewaxis[0], hand.origin );
// VectorMA( hand.origin, rxn_guny, cg.refdef.viewaxis[1], hand.origin );
// VectorMA( hand.origin, (rxn_gunz + fovOffset), cg.refdef.viewaxis[2], hand.origin );
2001-05-06 20:50:27 +00:00
//Blaze end:
//Blaze: Use above code instead
//Elder: reused so Cloud can adjust the models manually
VectorMA( hand.origin, cg_gun_x.value, cg.refdef.viewaxis[0], hand.origin );
VectorMA( hand.origin, cg_gun_y.value, cg.refdef.viewaxis[1], hand.origin );
VectorMA( hand.origin, (cg_gun_z.value+fovOffset), cg.refdef.viewaxis[2], hand.origin );
2001-05-06 20:50:27 +00:00
AnglesToAxis( angles, hand.axis );
//Elder: temp hack
2001-10-30 23:23:04 +00:00
// map torso animations to weapon animations
/*
if ( cg_gun_frame.integer ||
ps->weapon == WP_PISTOL ||
ps->weapon == WP_M3 ||
ps->weapon == WP_HANDCANNON ||
ps->weapon == WP_SSG3000 ||
ps->weapon == WP_M4 ||
ps->weapon == WP_MP5 ||
ps->weapon == WP_AKIMBO ||
ps->weapon == WP_GRENADE ||
ps->weapon == WP_KNIFE ) {
*/
// development tool
hand.frame = hand.oldframe = cg_gun_frame.integer;
hand.backlerp = 0;
/*
} else {
// get clientinfo for animation map
ci = &cgs.clientinfo[ cent->currentState.clientNum ];
hand.frame = CG_MapTorsoToWeaponFrame( ci, cent->pe.torso.frame );
hand.oldframe = CG_MapTorsoToWeaponFrame( ci, cent->pe.torso.oldFrame );
hand.backlerp = cent->pe.torso.backlerp;
}
*/
2001-05-06 20:50:27 +00:00
2001-07-28 12:32:54 +00:00
hand.hModel = weapon->handsModel;
2001-05-06 20:50:27 +00:00
hand.renderfx = RF_DEPTHHACK | RF_FIRST_PERSON | RF_MINLIGHT;
// add everything onto the hand
CG_AddPlayerWeapon( &hand, ps, &cg.predictedPlayerEntity, ps->persistant[PERS_TEAM] );
}
/*
==============================================================================
WEAPON SELECTION
==============================================================================
*/
/*
===================
CG_DrawWeaponSelect
===================
*/
void CG_DrawWeaponSelect( void ) {
int i;
int bits;
int count;
int x, y, w;
char *name;
float *color;
// don't display if dead
if ( cg.predictedPlayerState.stats[STAT_HEALTH] <= 0 ) {
return;
}
color = CG_FadeColor( cg.weaponSelectTime, WEAPON_SELECT_TIME );
if ( !color ) {
return;
}
trap_R_SetColor( color );
// showing weapon select clears pickup item display, but not the blend blob
cg.itemPickupTime = 0;
// count the number of weapons owned
bits = cg.snap->ps.stats[ STAT_WEAPONS ];
count = 0;
for ( i = 1 ; i < 16 ; i++ ) {
if ( bits & ( 1 << i ) ) {
count++;
}
}
x = 320 - count * 20;
y = 380;
for ( i = 1 ; i < 16 ; i++ ) {
if ( !( bits & ( 1 << i ) ) ) {
continue;
}
CG_RegisterWeapon( i );
// draw weapon icon
CG_DrawPic( x, y, 32, 32, cg_weapons[i].weaponIcon );
// draw selection marker
if ( i == cg.weaponSelect ) {
CG_DrawPic( x-4, y-4, 40, 40, cgs.media.selectShader );
}
// no ammo cross on top
if ( !cg.snap->ps.ammo[ i ] ) {
CG_DrawPic( x, y, 32, 32, cgs.media.noammoShader );
}
x += 40;
}
// draw the selected name
if ( cg_weapons[ cg.weaponSelect ].item ) {
name = cg_weapons[ cg.weaponSelect ].item->pickup_name;
if ( name ) {
w = CG_DrawStrlen( name ) * BIGCHAR_WIDTH;
x = ( SCREEN_WIDTH - w ) / 2;
CG_DrawBigStringColor(x, y - 22, name, color);
}
}
trap_R_SetColor( NULL );
}
/*
===============
CG_WeaponSelectable
===============
*/
static qboolean CG_WeaponSelectable( int i ) {
//Elder: never switch to empty grenades/knives
if ( !cg.snap->ps.ammo[i] && (i == WP_GRENADE || i == WP_KNIFE) ) {
2001-05-06 20:50:27 +00:00
return qfalse;
}
//Blaze: Check the amount of clips too for the weapon rotate code
//Elder: this won't work plus AQ2 lets you select empty weapons
//if ( !cg.snap->ps.ammo[i] && !cg.snap->ps.stats[STAT_CLIPS] ) {
//return qfalse;
//}
2001-05-06 20:50:27 +00:00
if ( ! (cg.snap->ps.stats[ STAT_WEAPONS ] & ( 1 << i ) ) ) {
return qfalse;
}
return qtrue;
}
/*
===============
CG_NextWeapon_f
===============
*/
void CG_NextWeapon_f( void ) {
int i;
int original;
if ( !cg.snap ) {
return;
}
if ( cg.snap->ps.pm_flags & PMF_FOLLOW ) {
return;
}
// if we are going into the intermission, don't do anything
if ( cg.intermissionStarted ) {
return;
}
//Elder: added
if ( (cg.snap->ps.stats[STAT_RQ3] & RQ3_BANDAGE_WORK) == RQ3_BANDAGE_WORK) {
CG_Printf("You are too busy bandaging...\n");
return;
}
//Elder: don't allow weapon switching when in the middle of bursts
if (cg.snap->ps.stats[STAT_BURST] > 0)
return;
//Elder: in the middle of firing, reloading or weapon-switching
/*
if (cg.snap->ps.weaponstate == WEAPON_RELOADING && cg.snap->ps.weaponTime > 0) {
return;
}*/
//Elder: added
//cg.zoomed = qfalse;
//cg.zoomLevel = 0;
2001-05-06 20:50:27 +00:00
cg.weaponSelectTime = cg.time;
original = cg.weaponSelect;
for ( i = 0 ; i < 16 ; i++ ) {
cg.weaponSelect++;
if ( cg.weaponSelect == 16 ) {
cg.weaponSelect = 0;
}
//Blaze: Allow full Cycle
//if ( cg.weaponSelect == WP_GAUNTLET ) {
// continue; // never cycle to gauntlet
//}
if ( CG_WeaponSelectable( cg.weaponSelect ) ) {
break;
}
}
if ( i == 16 ) {
cg.weaponSelect = original;
}
else {
CG_RQ3_Zoom1x();
trap_SendClientCommand("unzoom");
}
2001-05-06 20:50:27 +00:00
}
/*
===============
CG_PrevWeapon_f
===============
*/
void CG_PrevWeapon_f( void ) {
int i;
int original;
if ( !cg.snap ) {
return;
}
if ( cg.snap->ps.pm_flags & PMF_FOLLOW ) {
return;
}
// if we are going into the intermission, don't do anything
if ( cg.intermissionStarted ) {
return;
}
//Elder: added
if ( (cg.snap->ps.stats[STAT_RQ3] & RQ3_BANDAGE_WORK) == RQ3_BANDAGE_WORK) {
CG_Printf("You are too busy bandaging...\n");
return;
}
//Elder: don't allow weapon switching when in the middle of bursts
if (cg.snap->ps.stats[STAT_BURST] > 0)
return;
//Elder: in the middle of firing, reloading or weapon-switching
//cg.snap->ps.weaponstate == WEAPON_RELOADING when it's in
/*
if (cg.snap->ps.weaponstate == WEAPON_RELOADING && cg.snap->ps.weaponTime > 0) {
return;
}
*/
//Elder: added
//cg.zoomed = qfalse;
//cg.zoomLevel = 0;
2001-05-06 20:50:27 +00:00
cg.weaponSelectTime = cg.time;
original = cg.weaponSelect;
for ( i = 0 ; i < 16 ; i++ ) {
cg.weaponSelect--;
if ( cg.weaponSelect == -1 ) {
cg.weaponSelect = 15;
}
//Blaze: Allow full Cycle
//if ( cg.weaponSelect == WP_GAUNTLET ) {
// continue; // never cycle to gauntlet
//}
if ( CG_WeaponSelectable( cg.weaponSelect ) ) {
break;
}
}
if ( i == 16 ) {
cg.weaponSelect = original;
}
else {
CG_RQ3_Zoom1x();
trap_SendClientCommand("unzoom");
}
2001-05-06 20:50:27 +00:00
}
/*
===============
Added by Elder
CG_SpecialWeapon_f
===============
*/
void CG_SpecialWeapon_f( void ) {
int i;
int original;
if ( !cg.snap ) {
//CG_Printf("No snapshot: normally exiting\n");
return;
}
if ( cg.snap->ps.pm_flags & PMF_FOLLOW ) {
return;
}
// if we are going into the intermission, don't do anything
if ( cg.intermissionStarted ) {
return;
}
//Elder: added
if ( (cg.snap->ps.stats[STAT_RQ3] & RQ3_BANDAGE_WORK) == RQ3_BANDAGE_WORK) {
CG_Printf("You are too busy bandaging...\n");
return;
}
//Elder: don't allow weapon switching when in the middle of bursts
if (cg.snap->ps.stats[STAT_BURST] > 0)
return;
//Elder: in the middle of firing, reloading or weapon-switching
//cg.snap->ps.weaponstate == WEAPON_RELOADING when it's in
/*
if (cg.snap->ps.weaponstate == WEAPON_RELOADING && cg.snap->ps.weaponTime > 0) {
return;
}
*/
cg.weaponSelectTime = cg.time;
original = cg.weaponSelect;
for ( i = 0 ; i < 16 ; i++ ) {
cg.weaponSelect++;
if ( cg.weaponSelect == 16 ) {
cg.weaponSelect = 0;
}
//Skip normal weapons
switch (cg.weaponSelect) {
case WP_PISTOL:
case WP_AKIMBO:
case WP_KNIFE:
case WP_GRENADE:
continue;
break;
}
if ( CG_WeaponSelectable( cg.weaponSelect ) ) {
break;
}
}
// FIXME: 15 because of the stupid continue I used
if ( i >= 15 ) {
cg.weaponSelect = original;
}
else {
CG_RQ3_Zoom1x();
trap_SendClientCommand("unzoom");
}
}
//Elder: for returning to the zoom state in ps stats
void CG_RQ3_SyncZoom ( void ) {
if ( (cg.snap->ps.stats[STAT_RQ3] & RQ3_ZOOM_LOW) == RQ3_ZOOM_LOW &&
(cg.snap->ps.stats[STAT_RQ3] & RQ3_ZOOM_MED) == RQ3_ZOOM_MED ) {
//CG_Printf("Zoomed to 6x\n");
cg.zoomLevel = RQ3_ZOOM_LOW|RQ3_ZOOM_MED;
}
else if ( (cg.snap->ps.stats[STAT_RQ3] & RQ3_ZOOM_LOW) == RQ3_ZOOM_LOW) {
//CG_Printf("Zoomed to 2x\n");
cg.zoomLevel = RQ3_ZOOM_LOW;
}
else if ( (cg.snap->ps.stats[STAT_RQ3] & RQ3_ZOOM_MED) == RQ3_ZOOM_MED) {
//CG_Printf("Zoomed to 4x\n");
cg.zoomLevel = RQ3_ZOOM_MED;
}
else {
//CG_Printf("Zoomed out\n");
cg.zoomLevel = 0;
}
}
//Elder: save zoom level and do any other necessary housekeeping
void CG_RQ3_SaveZoomLevel() {
cg.lastZoomLevel = cg.zoomLevel;
}
/*
void CG_RQ3_SyncZoom2 ( void ) {
int statZoom;
if ( (cg.snap->ps.stats[STAT_RQ3] & RQ3_ZOOM_LOW) == RQ3_ZOOM_LOW &&
(cg.snap->ps.stats[STAT_RQ3] & RQ3_ZOOM_MED) == RQ3_ZOOM_MED ) {
statZoom = RQ3_ZOOM_LOW|RQ3_ZOOM_MED;
}
else if ( (cg.snap->ps.stats[STAT_RQ3] & RQ3_ZOOM_LOW) == RQ3_ZOOM_LOW) {
statZoom = RQ3_ZOOM_LOW;
}
else if ( (cg.snap->ps.stats[STAT_RQ3] & RQ3_ZOOM_MED) == RQ3_ZOOM_MED) {
statZoom = RQ3_ZOOM_MED;
}
else {
statZoom = 0;
}
if (statZoom != cg.zoomLevel)
cg.zoomLevel = statZoom;
}*/
//Elder: This should identical to the portion inside
//the weapon function in g_cmds.c
void CG_RQ3_Zoom( void ) {
//Elder: reworked SSG zoom
if (cg.snap->ps.weapon == WP_SSG3000)
{
CG_RQ3_SaveZoomLevel();
if ( (cg.zoomLevel & RQ3_ZOOM_LOW) == RQ3_ZOOM_LOW &&
(cg.zoomLevel & RQ3_ZOOM_MED) == RQ3_ZOOM_MED ) {
//Elder: zoom 1x
cg.zoomLevel &= ~RQ3_ZOOM_LOW;
cg.zoomLevel &= ~RQ3_ZOOM_MED;
}
else if ( (cg.zoomLevel & RQ3_ZOOM_MED) == RQ3_ZOOM_MED) {
//Elder: zoom 6x
cg.zoomLevel |= RQ3_ZOOM_LOW;
}
else if ( (cg.zoomLevel & RQ3_ZOOM_LOW) == RQ3_ZOOM_LOW) {
//Elder: zoom 4x
cg.zoomLevel |= RQ3_ZOOM_MED;
cg.zoomLevel &= ~RQ3_ZOOM_LOW;
}
else {
//Elder: zoom 2x
cg.zoomLevel |= RQ3_ZOOM_LOW;
}
cg.zoomTime = cg.time;
}
/*
2001-05-06 20:50:27 +00:00
if(cg.snap->ps.weapon==WP_SSG3000) {
cg.zoomLevel++;
if (cg.zoomLevel == 4) {
cg.zoomed = qfalse;
cg.zoomLevel = 0;
CG_Printf("Zoomed out\n");
}
else {
cg.zoomed = qtrue;
CG_Printf("Zoomed to %dx\n", cg.zoomLevel * 2);
}
cg.zoomTime = cg.time;
}
*/
}
//Elder: reset locally and send server message
void CG_RQ3_Zoom1x () {
if (cg_RQ3_ssgZoomAssist.integer)
{
cg.lastZoomLevel = cg.zoomLevel;
cg.zoomLevel = 0;
cg.zoomTime = cg.time;
}
//trap_SendClientCommand("unzoom");
}
int CG_RQ3_GetGrenadeMode()
{
int grenMode = 0;
if ( (cg.snap->ps.persistant[PERS_WEAPONMODES] & RQ3_GRENSHORT) == RQ3_GRENSHORT)
grenMode |= RQ3_GRENSHORT;
if ( (cg.snap->ps.persistant[PERS_WEAPONMODES] & RQ3_GRENMED) == RQ3_GRENMED)
grenMode |= RQ3_GRENMED;
2001-05-06 20:50:27 +00:00
return grenMode;
2001-05-06 20:50:27 +00:00
}
//Print grenade mode message
void CG_RQ3_GrenadeMode()
{
//Print a message for the next mode in line
if ( (cg.snap->ps.persistant[PERS_WEAPONMODES] & RQ3_GRENSHORT) == RQ3_GRENSHORT &&
(cg.snap->ps.persistant[PERS_WEAPONMODES] & RQ3_GRENMED) == RQ3_GRENMED ) {
CG_Printf("Prepared to make a short range throw\n");
//cg.grenadeMode = RQ3_GRENSHORT|RQ3_GRENMED;
}
else if ( (cg.snap->ps.persistant[PERS_WEAPONMODES] & RQ3_GRENSHORT) == RQ3_GRENSHORT) {
CG_Printf("Prepared to make a medium range throw\n");
//cg.grenadeMode = RQ3_GRENSHORT;
}
else if ( (cg.snap->ps.persistant[PERS_WEAPONMODES] & RQ3_GRENMED) == RQ3_GRENMED) {
CG_Printf("Prepared to make a long range throw\n");
//cg.grenadeMode = RQ3_GRENMED;
}
}
2001-05-06 20:50:27 +00:00
/*
===============
CG_Weapon_f
Elder: does local stuff, then proceeds to server
Used with Q3's internal weapon switching "e.g. weapon 10"
2001-05-06 20:50:27 +00:00
===============
*/
void CG_Weapon_f( void ) {
int num;
2001-05-06 20:50:27 +00:00
if ( !cg.snap ) {
//CG_Printf("No snapshot: normally exiting\n");
2001-05-06 20:50:27 +00:00
return;
}
// if we are going into the intermission, don't do anything
if ( cg.intermissionStarted ) {
return;
}
//Elder: added to prevent weapon switching while bandaging
if ( (cg.snap->ps.stats[STAT_RQ3] & RQ3_BANDAGE_WORK) == RQ3_BANDAGE_WORK) {
CG_Printf("You are too busy bandaging...\n");
return;
}
//Elder: don't allow weapon switching when in the middle of bursts
if (cg.snap->ps.stats[STAT_BURST] > 0)
return;
/*
//Elder: in the middle of firing, reloading or weapon-switching
if (cg.snap->ps.weaponTime > 0 || cg.snap->ps.stats[STAT_RELOADTIME] > 0) {
return;
}
*/
///Elder: spectator?
2001-05-06 20:50:27 +00:00
if ( cg.snap->ps.pm_flags & PMF_FOLLOW ) {
return;
}
// Hawkins (give 'weapon' dual meaning)
if ( trap_Argc() == 1 ) {
//Elder: in the middle of firing, reloading or weapon-switching
if (cg.snap->ps.weaponTime > 0 || cg.snap->ps.stats[STAT_RELOADTIME] > 0) {
return;
}
if (cg.snap->ps.weapon == WP_SSG3000) {
2002-02-23 18:07:46 +00:00
//Slicer- Changing Sniper stuff- some to game.
/* trap_S_StartLocalSound( cgs.media.lensSound, CHAN_ITEM);
if (cg_RQ3_ssgZoomAssist.integer)
CG_RQ3_Zoom();
2002-02-23 18:07:46 +00:00
*/
//Let's sound it..
trap_S_StartLocalSound( cgs.media.lensSound, CHAN_ITEM);
//Save the time..
cg.zoomTime = cg.time;
}
else if (cg.snap->ps.weapon == WP_GRENADE)
{
CG_RQ3_GrenadeMode();
}
2001-10-30 23:23:04 +00:00
// only play "click" sound for M4/MP5/Pistol
else if (cg.snap->ps.weapon == WP_M4 || cg.snap->ps.weapon == WP_MP5 ||
cg.snap->ps.weapon == WP_PISTOL )
{
trap_S_StartLocalSound( cgs.media.weapToggleSound, CHAN_ITEM);
}
trap_SendClientCommand("weapon");
//Elder: added to get out of function at this point
return;
}
2001-05-06 20:50:27 +00:00
num = atoi( CG_Argv( 1 ) );
if ( num < 1 || num > 15 ) {
return;
}
//Elder: this point on is the regular Q3 weapon function - weird
2001-05-06 20:50:27 +00:00
cg.weaponSelectTime = cg.time;
if ( ! ( cg.snap->ps.stats[STAT_WEAPONS] & ( 1 << num ) ) ) {
return; // don't have the weapon
}
//Elder: If re-selecting SSG - leave function
if ( (1 << num) == (1 << WP_SSG3000) && cg.snap->ps.weapon == WP_SSG3000) {
//CG_Printf("Selecting SSG\n");
return;
}
//cg.zoomed = qfalse;
//cg.zoomLevel = 0;
CG_RQ3_Zoom1x();
trap_SendClientCommand("unzoom");
2001-05-06 20:50:27 +00:00
cg.weaponSelect = num;
}
/*
===================
CG_OutOfAmmoChange
The current weapon has just run out of ammo
===================
*/
void CG_OutOfAmmoChange( void ) {
//int i;
2001-05-06 20:50:27 +00:00
if (!cg.snap)
return;
//Select the pistol when we run out of grenades or knives
if (cg.snap->ps.weapon == WP_KNIFE || cg.snap->ps.weapon == WP_GRENADE) {
cg.weaponSelectTime = cg.time;
cg.weaponSelect = WP_PISTOL;
}
/* Elder: disable auto-switch
2001-05-06 20:50:27 +00:00
cg.weaponSelectTime = cg.time;
//Elder: we'll have to change this to
//a) show the empty gun animation if gun's ammo = 0
//b) NOT auto-switch to another weapon
2001-05-06 20:50:27 +00:00
for ( i = 15 ; i > 0 ; i-- ) {
if ( CG_WeaponSelectable( i ) ) {
cg.weaponSelect = i;
break;
}
}
*/
2001-05-06 20:50:27 +00:00
}
/*
===================================================================================================
WEAPON EVENTS
===================================================================================================
*/
/*
================
CG_FireWeapon
Caused by an EV_FIRE_WEAPON event
================
*/
void CG_FireWeapon( centity_t *cent, int weapModification ) {
2001-05-06 20:50:27 +00:00
entityState_t *ent;
int c, i;
2001-05-06 20:50:27 +00:00
weaponInfo_t *weap;
ent = &cent->currentState;
if ( ent->weapon == WP_NONE ) {
return;
}
if ( ent->weapon >= WP_NUM_WEAPONS ) {
CG_Error( "CG_FireWeapon: ent->weapon >= WP_NUM_WEAPONS" );
return;
}
2001-05-06 20:50:27 +00:00
weap = &cg_weapons[ ent->weapon ];
//Elder: added to supress burst mode flashes + sounds when 'predicting'
// M4
if ( cg.snap->ps.weapon == WP_M4 &&
(cg.snap->ps.persistant[PERS_WEAPONMODES] & RQ3_M4MODE) == RQ3_M4MODE &&
cg.snap->ps.stats[STAT_BURST] > 2 )
{
return;
}
// MP5
if ( cg.snap->ps.weapon == WP_MP5 &&
(cg.snap->ps.persistant[PERS_WEAPONMODES] & RQ3_MP5MODE) == RQ3_MP5MODE &&
cg.snap->ps.stats[STAT_BURST] > 2 ) {
return;
}
// MK23
if ( cg.snap->ps.weapon == WP_PISTOL &&
(cg.snap->ps.persistant[PERS_WEAPONMODES] & RQ3_MK23MODE) == RQ3_MK23MODE &&
cg.snap->ps.stats[STAT_BURST] > 0 ) {
return;
}
2001-12-17 15:06:41 +00:00
// KNIFE
// NicEass: This is if the knife slash is not the first (2,3,4,5)
if (weapModification == RQ3_WPMOD_KNIFENOMARK) {
return;
}
/*if ( cg.snap->ps.weapon == WP_KNIFE ) // { && cg.snap->ps.stats[STAT_BURST] > 0 ) {
return;
}*/
2001-05-06 20:50:27 +00:00
// mark the entity as muzzle flashing, so when it is added it will
// append the flash to the weapon model
if (weapModification != RQ3_WPMOD_SILENCER)
{
cent->muzzleFlashTime = cg.time;
}
// Elder: choose alternate muzzle flashes for 3rd-person views
if (ent->weapon == WP_AKIMBO) {
if (cg.akimboFlash) {
cg.akimboFlash = 0;
} else {
cg.akimboFlash = 1;
}
}
2001-05-06 20:50:27 +00:00
// lightning gun only does this this on initial press
//Blaze: no more Lighting gun
/*
if ( ent->weapon == WP_LIGHTNING ) {
if ( cent->pe.lightningFiring ) {
return;
}
}
*/
// play quad sound if needed
if ( cent->currentState.powerups & ( 1 << PW_QUAD ) ) {
trap_S_StartSound (NULL, cent->currentState.number, CHAN_ITEM, cgs.media.quadSound );
}
//Elder: silencer stuff
if (weapModification == RQ3_WPMOD_SILENCER)
{
trap_S_StartSound( NULL, ent->number, CHAN_WEAPON, cgs.media.silencerSound );
}
2001-12-17 15:06:41 +00:00
else// if (cg.snap->ps.weapon != WP_KNIFE)
{
2001-07-27 00:21:23 +00:00
// play a sound
for ( c = 0 ; c < 4 ; c++ ) {
if ( !weap->flashSound[c] ) {
break;
}
2001-05-06 20:50:27 +00:00
}
2001-07-27 00:21:23 +00:00
if ( c > 0 ) {
c = rand() % c;
if ( weap->flashSound[c] )
{
trap_S_StartSound( NULL, ent->number, CHAN_WEAPON, weap->flashSound[c] );
}
2001-05-06 20:50:27 +00:00
}
}
//Elder: TODO: eject sync with animation for M3 and only eject for HC when reloading
2001-05-06 20:50:27 +00:00
// do brass ejection
if ( weap->ejectBrassFunc && cg_brassTime.integer > 0 ) {
weap->ejectBrassFunc( cent );
}
/*
// MK23
//Calculate the kick angles
for (i=1 ; i<3 ; i++)
{
ent->client->kick_origin[i] = crandom() * 0.35;
ent->client->kick_angles[i] = crandom() * 0.7;
}
ent->client->kick_origin[0] = crandom() * 0.35;
// Akimbo
//Calculate the kick angles
for (i=1 ; i<3 ; i++)
{
ent->client->kick_origin[i] = crandom() * 0.25;
ent->client->kick_angles[i] = crandom() * 0.5;
}
ent->client->kick_origin[0] = crandom() * 0.35;
// Handcannon
AngleVectors (ent->client->v_angle, forward, right, NULL);
VectorScale (forward, -2, ent->client->kick_origin);
ent->client->kick_angles[0] = -2;
// Knives
AngleVectors (ent->client->v_angle, forward, right, NULL);
VectorScale (forward, -2, ent->client->kick_origin);
ent->client->kick_angles[0] = -2;
// M3
AngleVectors (ent->client->v_angle, forward, right, NULL);
VectorScale (forward, -2, ent->client->kick_origin);
ent->client->kick_angles[0] = -2;
// M4 -- we already add the ride-up angles in pmove
//Calculate the kick angles
for (i=1 ; i<3 ; i++)
{
ent->client->kick_origin[i] = crandom() * 0.25;
ent->client->kick_angles[i] = crandom() * 0.5;
}
ent->client->kick_origin[0] = crandom() * 0.35;
// MP5
//Calculate the kick angles
for (i=1 ; i<3 ; i++)
{
ent->client->kick_origin[i] = crandom() * 0.25;
ent->client->kick_angles[i] = crandom() * 0.5;
}
ent->client->kick_origin[0] = crandom() * 0.35;
// SSG3000 has no kick
*/
// View kicks -- note this doesn't affect aim which is handled on the server-side
// even though it probably should
if (ent->clientNum == cg.snap->ps.clientNum)
{
vec3_t forward;
switch ( cg.snap->ps.weapon )
{
case WP_KNIFE:
AngleVectors(cg.predictedPlayerState.viewangles, forward, NULL, NULL);
VectorScale ( forward, -2, cg.kick_origin );
cg.kick_angles[0] = -2;
cg.kick_time = cg.time;
cg.kick_duration = 200;
break;
case WP_PISTOL:
for (i = 0; i < 3; i++)
{
cg.kick_origin[i] = crandom() * 0.35f;
cg.kick_angles[i] = crandom() * 0.7f;
}
cg.kick_origin[0] = crandom() * 0.35f;
cg.kick_time = cg.time;
break;
case WP_M3:
AngleVectors(cg.predictedPlayerState.viewangles, forward, NULL, NULL);
VectorScale ( forward, -2, cg.kick_origin );
cg.kick_angles[0] = -2;
cg.kick_time = cg.time;
break;
case WP_M4:
for (i = 0; i < 3; i++)
{
cg.kick_origin[i] = crandom() * 0.25f;
cg.kick_angles[i] = crandom() * 0.5f;
}
cg.kick_origin[0] = crandom() * 0.35f;
cg.kick_time = cg.time;
break;
case WP_MP5:
for (i = 0; i < 3; i++)
{
cg.kick_origin[i] = crandom() * 0.25f;
cg.kick_angles[i] = crandom() * 0.5f;
}
cg.kick_origin[0] = crandom() * 0.35f;
cg.kick_time = cg.time;
break;
case WP_HANDCANNON:
AngleVectors(cg.predictedPlayerState.viewangles, forward, NULL, NULL);
VectorScale ( forward, -2, cg.kick_origin );
cg.kick_angles[0] = -2;
cg.kick_time = cg.time;
break;
case WP_AKIMBO:
for (i = 0; i < 3; i++)
{
cg.kick_origin[i] = crandom() * 0.25f;
cg.kick_angles[i] = crandom() * 0.5f;
}
cg.kick_origin[0] = crandom() * 0.35f;
cg.kick_time = cg.time;
break;
}
}
2001-05-06 20:50:27 +00:00
}
/*
===============
CG_ShrapnelSpark
Added by Elder
Modified tracer code
I really don't know what's going on half the time here :)
===============
*/
static void CG_ShrapnelSpark( vec3_t source, vec3_t dest, float width, float length ) {
vec3_t forward, right;
polyVert_t verts[4];
vec3_t line;
float len, begin, end;
vec3_t start, finish;
//vec3_t midpoint;
// tracer
VectorSubtract( dest, source, forward );
len = VectorNormalize( forward );
// start at least a little ways from the muzzle
//if ( len < 10 ) {
//return;
//}
begin = crandom() * 8;
end = begin + length;
if ( end > len ) {
end = len;
}
VectorMA( source, begin, forward, start );
VectorMA( source, end, forward, finish );
line[0] = DotProduct( forward, cg.refdef.viewaxis[1] );
line[1] = DotProduct( forward, cg.refdef.viewaxis[2] );
VectorScale( cg.refdef.viewaxis[1], line[1], right );
VectorMA( right, -line[0], cg.refdef.viewaxis[2], right );
VectorNormalize( right );
VectorMA( finish, width, right, verts[0].xyz );
verts[0].st[0] = 0;
verts[0].st[1] = 1;
verts[0].modulate[0] = 255;
verts[0].modulate[1] = 255;
verts[0].modulate[2] = 255;
verts[0].modulate[3] = 255;
VectorMA( finish, -width, right, verts[1].xyz );
verts[1].st[0] = 1;
verts[1].st[1] = 0;
verts[1].modulate[0] = 255;
verts[1].modulate[1] = 255;
verts[1].modulate[2] = 255;
verts[1].modulate[3] = 255;
VectorMA( start, -width, right, verts[2].xyz );
verts[2].st[0] = 1;
verts[2].st[1] = 1;
verts[2].modulate[0] = 255;
verts[2].modulate[1] = 255;
verts[2].modulate[2] = 255;
verts[2].modulate[3] = 255;
VectorMA( start, width, right, verts[3].xyz );
verts[3].st[0] = 0;
verts[3].st[1] = 0;
verts[3].modulate[0] = 255;
verts[3].modulate[1] = 255;
verts[3].modulate[2] = 255;
verts[3].modulate[3] = 255;
trap_R_AddPolyToScene( cgs.media.tracerShader, 4, verts );
//midpoint[0] = ( start[0] + finish[0] ) * 0.5;
//midpoint[1] = ( start[1] + finish[1] ) * 0.5;
//midpoint[2] = ( start[2] + finish[2] ) * 0.5;
// add the tracer sound
//trap_S_StartSound( midpoint, ENTITYNUM_WORLD, CHAN_AUTO, cgs.media.tracerSound );
}
2001-05-06 20:50:27 +00:00
/*
=================
CG_MissileHitWall
Caused by an EV_MISSILE_MISS event, or directly by local bullet tracing
=================
*/
void CG_MissileHitWall( int weapon, int clientNum, vec3_t origin,
2001-12-17 15:06:41 +00:00
vec3_t dir, vec3_t viewdir, impactSound_t soundType, int weapModification ) {
2001-05-06 20:50:27 +00:00
qhandle_t mod;
qhandle_t mark;
qhandle_t shader;
sfxHandle_t sfx;
float radius;
float light;
vec3_t lightColor;
localEntity_t *le;
int r;
qboolean alphaFade;
qboolean isSprite;
int duration;
2001-08-18 20:01:21 +00:00
int angle;
//Elder: for impact smoke marks
localEntity_t *smokePuff;
vec3_t puffOrigin;
vec3_t puffOffset;
vec3_t puffDir;
int contentType;
//Elder: for impact sparks
2001-12-17 15:06:41 +00:00
vec3_t velocity, dir2;
int sparkCount;
2001-12-17 15:06:41 +00:00
float dot;
int i;
2001-05-06 20:50:27 +00:00
mark = 0;
radius = 32;
sfx = 0;
mod = 0;
shader = 0;
light = 0;
lightColor[0] = 1;
lightColor[1] = 1;
lightColor[2] = 0;
// set defaults
isSprite = qfalse;
duration = 600;
switch ( weapon ) {
2001-11-12 08:59:46 +00:00
// Elder: bullet weapons all fall under MK23 >:(
2001-10-30 23:23:04 +00:00
case WP_M4:
case WP_MP5:
case WP_PISTOL:
case WP_AKIMBO:
case WP_SSG3000:
2001-11-12 08:59:46 +00:00
duration = 160;
mod = cgs.media.hitSparkModel;
shader = cgs.media.hitSparkShader;
radius = 8;
2001-10-30 23:23:04 +00:00
r = rand() & 3;
2001-10-30 23:23:04 +00:00
if (soundType == IMPACTSOUND_METAL)
{
2001-11-08 02:37:11 +00:00
mark = cgs.media.metalMarkShader;
2001-10-30 23:23:04 +00:00
if ( r < 2 )
sfx = cgs.media.sfx_metalric1;
else if ( r == 2 )
sfx = cgs.media.sfx_metalric2;
else
sfx = cgs.media.sfx_metalric3;
}
else if (soundType == IMPACTSOUND_GLASS)
{
2001-11-08 02:37:11 +00:00
mark = cgs.media.glassMarkShader;
2001-10-30 23:23:04 +00:00
if ( r < 2 )
sfx = cgs.media.sfx_glassric1;
else if ( r == 2 )
sfx = cgs.media.sfx_glassric2;
else
sfx = cgs.media.sfx_glassric3;
}
else
{
2001-11-08 02:37:11 +00:00
mark = cgs.media.bulletMarkShader;
2001-10-30 23:23:04 +00:00
if ( r == 0 )
sfx = cgs.media.sfx_ric1;
else if ( r == 1 )
sfx = cgs.media.sfx_ric2;
else
sfx = cgs.media.sfx_ric3;
}
break;
2001-05-06 20:50:27 +00:00
2001-10-30 23:23:04 +00:00
/*
//Blaze: Reaction M4
case WP_M4:
mod = cgs.media.bulletFlashModel;
shader = cgs.media.bulletExplosionShader;
if (soundType == IMPACTSOUND_GLASS)
mark = cgs.media.glassMarkShader;
else
mark = cgs.media.bulletMarkShader;
r = rand() & 3;
if (soundType == IMPACTSOUND_METAL)
{
if ( r < 2 ) {
sfx = cgs.media.sfx_metalric1;
} else if ( r == 2 ) {
sfx = cgs.media.sfx_metalric2;
} else {
sfx = cgs.media.sfx_metalric3;
}
}
2001-10-30 23:23:04 +00:00
else if (soundType == IMPACTSOUND_GLASS)
{
if ( r < 2 ) {
sfx = cgs.media.sfx_glassric1;
} else if ( r == 2 ) {
sfx = cgs.media.sfx_glassric2;
} else {
sfx = cgs.media.sfx_glassric3;
}
}
2001-10-30 23:23:04 +00:00
else
{
if ( r < 2 ) {
sfx = cgs.media.sfx_ric1;
} else if ( r == 2 ) {
sfx = cgs.media.sfx_ric2;
} else {
sfx = cgs.media.sfx_ric3;
}
}
2001-05-06 20:50:27 +00:00
2001-10-30 23:23:04 +00:00
radius = 8;
break;
//Blaze: Reaction Pistol
case WP_PISTOL:
mod = cgs.media.bulletFlashModel;
shader = cgs.media.bulletExplosionShader;
if (soundType == IMPACTSOUND_GLASS)
mark = cgs.media.glassMarkShader;
else
mark = cgs.media.bulletMarkShader;
2001-05-06 20:50:27 +00:00
2001-10-30 23:23:04 +00:00
r = rand() & 3;
if (soundType == IMPACTSOUND_METAL)
{
if ( r < 2 ) {
sfx = cgs.media.sfx_metalric1;
} else if ( r == 2 ) {
sfx = cgs.media.sfx_metalric2;
} else {
sfx = cgs.media.sfx_metalric3;
}
}
2001-10-30 23:23:04 +00:00
else if (soundType == IMPACTSOUND_GLASS)
{
if ( r < 2 ) {
sfx = cgs.media.sfx_glassric1;
} else if ( r == 2 ) {
sfx = cgs.media.sfx_glassric2;
} else {
sfx = cgs.media.sfx_glassric3;
}
}
2001-10-30 23:23:04 +00:00
else
{
if ( r < 2 ) {
sfx = cgs.media.sfx_ric1;
} else if ( r == 2 ) {
sfx = cgs.media.sfx_ric2;
} else {
sfx = cgs.media.sfx_ric3;
}
}
2001-10-30 23:23:04 +00:00
radius = 8;
break;
case WP_SSG3000:
mod = cgs.media.bulletFlashModel;
shader = cgs.media.bulletExplosionShader;
if (soundType == IMPACTSOUND_GLASS)
mark = cgs.media.glassMarkShader;
else
mark = cgs.media.bulletMarkShader;
2001-05-06 20:50:27 +00:00
2001-10-30 23:23:04 +00:00
r = rand() & 3;
if (soundType == IMPACTSOUND_METAL)
{
if ( r < 2 ) {
sfx = cgs.media.sfx_metalric1;
} else if ( r == 2 ) {
sfx = cgs.media.sfx_metalric2;
} else {
sfx = cgs.media.sfx_metalric3;
}
}
2001-10-30 23:23:04 +00:00
else if (soundType == IMPACTSOUND_GLASS)
{
if ( r < 2 ) {
sfx = cgs.media.sfx_glassric1;
} else if ( r == 2 ) {
sfx = cgs.media.sfx_glassric2;
} else {
sfx = cgs.media.sfx_glassric3;
}
}
2001-10-30 23:23:04 +00:00
else
{
if ( r < 2 ) {
sfx = cgs.media.sfx_ric1;
} else if ( r == 2 ) {
sfx = cgs.media.sfx_ric2;
} else {
sfx = cgs.media.sfx_ric3;
}
}
2001-10-30 23:23:04 +00:00
radius = 8;
break;
case WP_AKIMBO:
mod = cgs.media.bulletFlashModel;
shader = cgs.media.bulletExplosionShader;
if (soundType == IMPACTSOUND_GLASS)
mark = cgs.media.glassMarkShader;
else
mark = cgs.media.bulletMarkShader;
2001-10-30 23:23:04 +00:00
r = rand() & 3;
if (soundType == IMPACTSOUND_METAL)
{
if ( r < 2 ) {
sfx = cgs.media.sfx_metalric1;
} else if ( r == 2 ) {
sfx = cgs.media.sfx_metalric2;
} else {
sfx = cgs.media.sfx_metalric3;
}
}
2001-10-30 23:23:04 +00:00
else if (soundType == IMPACTSOUND_GLASS)
{
if ( r < 2 ) {
sfx = cgs.media.sfx_glassric1;
} else if ( r == 2 ) {
sfx = cgs.media.sfx_glassric2;
} else {
sfx = cgs.media.sfx_glassric3;
}
}
2001-10-30 23:23:04 +00:00
else
{
if ( r < 2 ) {
sfx = cgs.media.sfx_ric1;
} else if ( r == 2 ) {
sfx = cgs.media.sfx_ric2;
} else {
sfx = cgs.media.sfx_ric3;
}
}
2001-10-30 23:23:04 +00:00
radius = 8;
break;
2001-10-30 23:23:04 +00:00
//Blaze: Reaction MP5
case WP_MP5:
mod = cgs.media.bulletFlashModel;
shader = cgs.media.bulletExplosionShader;
if (soundType == IMPACTSOUND_GLASS)
mark = cgs.media.glassMarkShader;
else
mark = cgs.media.bulletMarkShader;
2001-05-06 20:50:27 +00:00
2001-10-30 23:23:04 +00:00
r = rand() & 3;
if (soundType == IMPACTSOUND_METAL)
{
if ( r < 2 ) {
sfx = cgs.media.sfx_metalric1;
} else if ( r == 2 ) {
sfx = cgs.media.sfx_metalric2;
} else {
sfx = cgs.media.sfx_metalric3;
}
}
2001-10-30 23:23:04 +00:00
else if (soundType == IMPACTSOUND_GLASS)
{
if ( r < 2 ) {
sfx = cgs.media.sfx_glassric1;
} else if ( r == 2 ) {
sfx = cgs.media.sfx_glassric2;
} else {
sfx = cgs.media.sfx_glassric3;
}
}
2001-10-30 23:23:04 +00:00
else
{
if ( r == 0 ) {
sfx = cgs.media.sfx_ric1;
} else if ( r == 1 ) {
sfx = cgs.media.sfx_ric2;
} else {
sfx = cgs.media.sfx_ric3;
}
}
2001-10-30 23:23:04 +00:00
radius = 8;
break;
*/
2001-10-30 23:23:04 +00:00
case WP_M3:
case WP_HANDCANNON:
2001-11-12 08:59:46 +00:00
duration = 200;
mod = cgs.media.hitSparkModel;
shader = cgs.media.hitSparkShader;
//mod = cgs.media.bulletFlashModel;
//shader = cgs.media.bulletExplosionShader;
2001-10-30 23:23:04 +00:00
if (soundType == IMPACTSOUND_GLASS)
mark = cgs.media.glassMarkShader;
2001-11-11 07:23:16 +00:00
else if (soundType == IMPACTSOUND_METAL)
mark = cgs.media.metalMarkShader;
2001-10-30 23:23:04 +00:00
else
mark = cgs.media.bulletMarkShader;
sfx = 0;
radius = 4;
break;
case WP_GRENADE:
mod = cgs.media.dishFlashModel;
shader = cgs.media.grenadeExplosionShader;
sfx = cgs.media.sfx_rockexp;
mark = cgs.media.burnMarkShader;
radius = 96; //64
light = 350; //300
isSprite = qtrue;
break;
2001-10-30 23:23:04 +00:00
case WP_KNIFE:
/*
mod = cgs.media.dishFlashModel;
//shader = cgs.media.grenadeExplosionShader;
sfx = cgs.media.sfx_rockexp;
mark = cgs.media.burnMarkShader;
radius = 64;
light = 300;
isSprite = qtrue;
break;
*/
if (weapModification == RQ3_WPMOD_KNIFESLASH)
{
2001-11-12 08:59:46 +00:00
duration = 100;
mod = cgs.media.hitSparkModel;
shader = cgs.media.hitSparkShader;
//mod = cgs.media.bulletFlashModel;
//shader = cgs.media.bulletExplosionShader;
2001-10-30 23:23:04 +00:00
mark = cgs.media.slashMarkShader;
sfx = cgs.media.knifeClankSound;
radius = rand() % 4 + 6;
}
else
{
2001-11-12 08:59:46 +00:00
duration = 180;
mod = cgs.media.hitSparkModel;
shader = cgs.media.hitSparkShader;
//mod = cgs.media.bulletFlashModel;
//shader = cgs.media.bulletExplosionShader;
2001-10-30 23:23:04 +00:00
sfx = cgs.media.knifeClankSound;
}
break;
default:
break;
2001-05-06 20:50:27 +00:00
}
2001-10-30 23:23:04 +00:00
//
// ricochet sound
//
if (weapon == WP_KNIFE || weapon == WP_SSG3000 || weapon == WP_GRENADE ||
soundType == IMPACTSOUND_METAL )
i = 1;
else
//Elder: 90% of the time render a bullet ricochet sound
i = rand() % 10;
if ( sfx && i < 9) {
2001-05-06 20:50:27 +00:00
trap_S_StartSound( origin, ENTITYNUM_WORLD, CHAN_AUTO, sfx );
}
//
// create the explosion
//
2001-11-12 08:59:46 +00:00
if ( mod )
{
if ( weapon == WP_GRENADE )
{
le = CG_MakeExplosion( origin, dir,
2001-05-06 20:50:27 +00:00
mod, shader,
duration, isSprite );
2001-11-12 08:59:46 +00:00
le->light = light;
VectorCopy( lightColor, le->lightColor );
}
2001-11-14 05:01:18 +00:00
else if ( cg_RQ3_impactEffects.integer )
2001-11-12 08:59:46 +00:00
{
vec3_t temp, offsetDir;
byte color[4];
int flashCount;
float scale;
switch ( weapon )
{
case WP_MP5:
case WP_PISTOL:
case WP_AKIMBO:
flashCount = 3 + rand() % 3;
color[0] = 224;
color[1] = 180;
color[2] = 128;
color[3] = 192;
break;
case WP_M4:
case WP_KNIFE:
flashCount = 3 + rand() % 3;
color[0] = 192;
color[1] = 96;
color[2] = 64;
color[3] = 192;
2001-11-25 23:04:08 +00:00
break;
2001-11-12 08:59:46 +00:00
case WP_M3:
case WP_HANDCANNON:
flashCount = 2 + rand() % 3;
color[0] = 192;
color[1] = 96;
color[2] = 64;
color[3] = 192;
break;
case WP_SSG3000:
default:
flashCount = 4 + rand() % 3;
color[0] = 255;
color[1] = 224;
color[2] = 128;
color[3] = 192;
break;
}
2001-12-17 15:06:41 +00:00
// NiceAss: Direction of bullet affects sparks
VectorCopy(dir, dir2);
if (weapon != WP_KNIFE) {
dot = DotProduct( viewdir, dir2 );
VectorMA( viewdir, -2*dot, dir2, dir2 );
}
2001-11-25 23:04:08 +00:00
2001-11-12 08:59:46 +00:00
// Elder: should probably dump this into another function
for ( i = 0; i < flashCount; i++ )
{
// introduce variance
2001-12-17 15:06:41 +00:00
VectorCopy( dir2, temp );
2001-11-12 08:59:46 +00:00
scale = crandom() + 1.8f;
temp[0] += (crandom() * 0.4f) - 0.2f;
temp[1] += (crandom() * 0.4f) - 0.2f;
temp[2] += (crandom() * 0.4f) - 0.2f;
// save result
VectorCopy( temp, offsetDir );
VectorScale( temp, scale, temp );
2001-12-17 15:06:41 +00:00
le = CG_MakeExplosion( origin, dir2,
2001-11-12 08:59:46 +00:00
mod, shader,
duration, isSprite );
VectorMA( origin, scale + 0.5f, temp, le->refEntity.origin );
VectorCopy(temp, le->refEntity.axis[0]);
2001-11-25 23:04:08 +00:00
2001-11-12 08:59:46 +00:00
// set spark colour
le->refEntity.shaderRGBA[0] = color[0];
le->refEntity.shaderRGBA[1] = color[1];
le->refEntity.shaderRGBA[2] = color[2];
le->refEntity.shaderRGBA[3] = color[3];
2001-11-25 23:04:08 +00:00
2001-11-12 08:59:46 +00:00
// readjust behaviour
le->leType = LE_MOVE_SCALE_FADE;
le->pos.trType = TR_LINEAR;
le->pos.trTime = cg.time;
VectorCopy(le->refEntity.origin, le->pos.trBase);
VectorScale(offsetDir, 30, le->pos.trDelta);
2001-11-25 23:04:08 +00:00
2001-11-12 08:59:46 +00:00
le->light = light;
VectorCopy( lightColor, le->lightColor );
}
}
2001-05-06 20:50:27 +00:00
//Blaze: No railgun
/*
if ( weapon == WP_RAILGUN ) {
// colorize with client color
VectorCopy( cgs.clientinfo[clientNum].color1, le->color );
2001-05-06 20:50:27 +00:00
}
*/
}
//
// impact mark
//
alphaFade = (mark == cgs.media.energyMarkShader); // plasma fades alpha, all others fade color
2001-05-06 20:50:27 +00:00
//Blaze: No more railgun
2001-10-30 23:23:04 +00:00
/*
if ( weapon == WP_RAILGUN ) {
float *color;
2001-05-06 20:50:27 +00:00
// colorize with client color
2001-10-30 23:23:04 +00:00
color = cgs.clientinfo[clientNum].color;
CG_ImpactMark( mark, origin, dir, random()*360, color[0],color[1], color[2],1, alphaFade, radius, qfalse );
} else {
*/
2001-10-30 23:23:04 +00:00
// Elder: Our knife slashes are vertical
2001-08-18 20:01:21 +00:00
if (weapon == WP_KNIFE)
2001-10-30 23:23:04 +00:00
angle = random() * 90;
2001-08-18 20:01:21 +00:00
else
angle = random() * 360;
if ( mark )
CG_ImpactMark( mark, origin, dir, angle, 1,1,1,1, alphaFade, radius, qfalse );
2001-05-06 20:50:27 +00:00
//}
2001-10-30 23:23:04 +00:00
//
// impact visual effects
//
//Elder: 75% of the time render a smoke puff
i = rand() % 4;
if (cg_RQ3_impactEffects.integer && i < 3)
{
contentType = trap_CM_PointContents( origin, 0 );
// no puff in water
if ( !( contentType & CONTENTS_WATER ) ) {
switch ( weapon ) {
case WP_MP5:
case WP_M4:
case WP_PISTOL:
2001-10-30 23:23:04 +00:00
case WP_AKIMBO:
case WP_SSG3000:
puffDir[0] = 0;
puffDir[1] = 0;
puffDir[2] = 16;
VectorCopy(dir, puffOffset);
VectorNormalize(puffOffset);
VectorNegate(puffOffset, puffOffset);
VectorScale(puffOffset, 13, puffOffset);
VectorSubtract(origin, puffOffset, puffOrigin);
smokePuff = CG_SmokePuff( puffOrigin, puffDir,
(int)(random() * 100) % 4 + 13,
1, 1, 1, 0.25f,
650,
cg.time, 0,
LEF_PUFF_DONT_SCALE,
2001-10-30 23:23:04 +00:00
cgs.media.smokePuffAnimShader );
break;
}
}
}
// Elder: Spark effect for metal surfaces
if (cg_RQ3_impactEffects.integer && soundType == IMPACTSOUND_METAL)
{
if (weapon != WP_GRENADE)
{
if (weapon == WP_M3 || weapon == WP_HANDCANNON)
sparkCount = 5 + rand() % 5;
else if (weapon == WP_KNIFE)
sparkCount = 10 + rand() % 10;
else if (weapon == WP_SSG3000)
sparkCount = 25 + rand() % 20;
else
sparkCount = 15 + rand() % 15;
2001-12-17 15:06:41 +00:00
// NiceAss: Direction of bullet affects sparks
VectorCopy(dir, dir2);
if (weapon != WP_KNIFE) {
dot = DotProduct( viewdir, dir2 );
VectorMA( viewdir, -2*dot, dir2, dir2 );
}
// Generate the particles
for (i = 0; i < sparkCount; i++)
{
if (weapon == WP_KNIFE)
2001-12-17 15:06:41 +00:00
VectorScale(dir2, 50 + rand() % 10, velocity);
else
2001-12-17 15:06:41 +00:00
VectorScale(dir2, 150 + rand() % 30, velocity);
//random upwards sparks
if ( rand() % 5 < 1)
velocity[2] += 120 + rand() % 30;
velocity[0] += rand() % 50 - 25;
velocity[1] += rand() % 50 - 25;
2001-12-17 15:06:41 +00:00
// CG_ParticleSparks(origin, velocity, 100 + rand() % 120, 2, 2, -4, 1);
// NiceAss: Longer life, faster sparks == more eye candy! =)
CG_ParticleSparks(origin, velocity, 150 + rand() % 120, 2, 2, -5, 1);
}
}
}
// Elder: grenade explosion
if (cg_RQ3_impactEffects.integer && weapon == WP_GRENADE)
{
vec3_t shrapnelDest;
sparkCount = 60 + rand() % 10;
origin[2] += 32;
for (i = 0; i < sparkCount; i++)
{
VectorScale (dir, rand() % 200, velocity);
velocity[0] += rand() % 200 - 100;
velocity[1] += rand() % 200 - 100;
if (i % 8 == 7)
{
// Add shrapnel trace effect
VectorMA(origin, 0.7f, velocity, shrapnelDest);
CG_ShrapnelSpark(origin, shrapnelDest, 10, 280);
}
// Add sparks
CG_ParticleSparks(origin, velocity, 900 + rand() % 200, 5, 5, -2.5f, 3.5f);
}
// Add smoke puff
puffDir[0] = 0;
puffDir[1] = 0;
puffDir[2] = 20;
origin[2] -= 16;
smokePuff = CG_SmokePuff( origin, puffDir,
rand() % 12 + 48,
1, 1, 1, 0.4f,
1750,
cg.time, 0,
0,
cgs.media.smokePuffShader );
}
2001-05-06 20:50:27 +00:00
}
/*
=================
CG_MissileHitPlayer
=================
*/
void CG_MissileHitPlayer( int weapon, vec3_t origin, vec3_t dir, int entityNum ) {
CG_Bleed( origin, entityNum );
2001-05-06 20:50:27 +00:00
// some weapons will make an explosion with the blood, while
// others will just make the blood
//Blaze: None of these are valid
/*
switch ( weapon ) {
case WP_GRENADE_LAUNCHER:
case WP_ROCKET_LAUNCHER:
#ifdef MISSIONPACK
case WP_NAILGUN:
case WP_CHAINGUN:
case WP_PROX_LAUNCHER:
#endif
2001-08-18 20:01:21 +00:00
CG_MissileHitWall( weapon, 0, origin, dir, IMPACTSOUND_FLESH, 0 );
2001-05-06 20:50:27 +00:00
break;
default:
break;
}
*/
}
/*
============================================================================
SHOTGUN TRACING
============================================================================
*/
/*
================
CG_ShotgunPellet
Elder: added shellWeapon param
We are not going to show every HC impact
2001-05-06 20:50:27 +00:00
================
*/
static void CG_ShotgunPellet( vec3_t start, vec3_t end, int skipNum, int shellWeapon ) {
2001-05-06 20:50:27 +00:00
trace_t tr;
2001-12-17 15:06:41 +00:00
vec3_t viewDir;
2001-05-06 20:50:27 +00:00
int sourceContentType, destContentType;
CG_Trace( &tr, start, NULL, NULL, end, skipNum, MASK_SHOT );
2001-12-17 15:06:41 +00:00
CG_CalcViewDir2(start, end, viewDir);
2001-05-06 20:50:27 +00:00
sourceContentType = trap_CM_PointContents( start, 0 );
destContentType = trap_CM_PointContents( tr.endpos, 0 );
// FIXME: should probably move this cruft into CG_BubbleTrail
if ( sourceContentType == destContentType ) {
if ( sourceContentType & CONTENTS_WATER ) {
CG_BubbleTrail( start, tr.endpos, 32 );
}
} else if ( sourceContentType & CONTENTS_WATER ) {
trace_t trace;
trap_CM_BoxTrace( &trace, end, start, NULL, NULL, 0, CONTENTS_WATER );
CG_BubbleTrail( start, trace.endpos, 32 );
} else if ( destContentType & CONTENTS_WATER ) {
trace_t trace;
trap_CM_BoxTrace( &trace, start, end, NULL, NULL, 0, CONTENTS_WATER );
CG_BubbleTrail( tr.endpos, trace.endpos, 32 );
}
if ( tr.surfaceFlags & SURF_NOIMPACT ) {
return;
}
if ( cg_entities[tr.entityNum].currentState.eType == ET_PLAYER )
{
2001-05-06 20:50:27 +00:00
//Blaze: Changed WP_SHOTGUN to WP_M3
//Elder: don't display so many blood stains - so we can reduce slow down
cg.shellHits++;
if (cg.shellHits < MAX_SHELL_HITS)
CG_MissileHitPlayer( WP_M3, tr.endpos, tr.plane.normal, tr.entityNum );
}
else
{
if ( tr.surfaceFlags & SURF_NOIMPACT )
{
2001-05-06 20:50:27 +00:00
// SURF_NOIMPACT will not make a flame puff or a mark
return;
}
2001-10-30 23:23:04 +00:00
if ( (tr.surfaceFlags & SURF_METALSTEPS) || (tr.surfaceFlags & SURF_METAL2) )
{
2001-05-06 20:50:27 +00:00
//Blaze: Changed WP_SHOTGUN to WP_M3
if (shellWeapon == WP_M3)
2001-12-17 15:06:41 +00:00
CG_MissileHitWall( WP_M3, 0, tr.endpos, tr.plane.normal, viewDir, IMPACTSOUND_METAL, 0 );
else if (shellWeapon == WP_HANDCANNON && crandom() > 0.5)
{
//Elder: show only approximately every other impact mark
2001-12-17 15:06:41 +00:00
CG_MissileHitWall( WP_HANDCANNON, 0, tr.endpos, tr.plane.normal, viewDir, IMPACTSOUND_METAL, 0 );
}
}
else if ( tr.surfaceFlags & SURF_GLASS )
{
//Blaze: Changed WP_SHOTGUN to WP_M3
if (shellWeapon == WP_M3)
2001-12-17 15:06:41 +00:00
CG_MissileHitWall( WP_M3, 0, tr.endpos, tr.plane.normal, viewDir, IMPACTSOUND_GLASS, 0 );
else if (shellWeapon == WP_HANDCANNON && crandom() > 0.5)
{
//Elder: show only approximately every other impact mark
2001-12-17 15:06:41 +00:00
CG_MissileHitWall( WP_HANDCANNON, 0, tr.endpos, tr.plane.normal, viewDir, IMPACTSOUND_GLASS, 0 );
}
}
else
{
2001-10-30 23:23:04 +00:00
// Elder: By default, the M3 and HC will spark on all surfaces
// Blaze: Changed WP_SHOTGUN to WP_M3
if (shellWeapon == WP_M3)
2001-12-17 15:06:41 +00:00
CG_MissileHitWall( WP_M3, 0, tr.endpos, tr.plane.normal, viewDir, IMPACTSOUND_METAL, 0 );
else if (shellWeapon == WP_HANDCANNON && crandom() > 0.5)
{
//Elder: show only approximately every other impact mark
2001-12-17 15:06:41 +00:00
CG_MissileHitWall( WP_HANDCANNON, 0, tr.endpos, tr.plane.normal, viewDir, IMPACTSOUND_METAL, 0 );
}
2001-08-18 20:01:21 +00:00
//CG_MissileHitWall( WP_M3, 0, tr.endpos, tr.plane.normal, IMPACTSOUND_DEFAULT, 0 );
2001-05-06 20:50:27 +00:00
}
}
}
/*
================
CG_ShotgunPattern
Perform the same traces the server did to locate the
hit splashes
2001-05-06 20:50:27 +00:00
================
*/
static void CG_ShotgunPattern( vec3_t origin, vec3_t origin2, int otherEntNum, int shotType, int seed ) {
2001-05-06 20:50:27 +00:00
int i;
float r, u;
vec3_t end;
vec3_t forward, right, up;
int count;
int hc_multipler;
2001-05-06 20:50:27 +00:00
// derive the right and up vectors from the forward vector, because
// the client won't have any other information
VectorNormalize2( origin2, forward );
PerpendicularVector( right, forward );
CrossProduct( forward, right, up );
// generate the "random" spread pattern
//Elder: added
if (shotType == WP_M3)
count = DEFAULT_M3_COUNT;
else if (shotType == WP_HANDCANNON) {
count = DEFAULT_HANDCANNON_COUNT;
hc_multipler = 4;
}
else {
count = DEFAULT_HANDCANNON_COUNT;
hc_multipler = 5;
}
cg.shellHits = 0;
for ( i = 0 ; i < count ; i++ ) {
if (shotType == WP_M3)
2001-05-06 20:50:27 +00:00
{
r = Q_crandom(&seed) * DEFAULT_M3_HSPREAD * 16;
u = Q_crandom(&seed) * DEFAULT_M3_VSPREAD * 16;
//r = crandom() * DEFAULT_M3_HSPREAD * 16;
//u = crandom() * DEFAULT_M3_VSPREAD * 16;
//r = crandom() * DEFAULT_SHOTGUN_SPREAD * 16;
//u = crandom() * DEFAULT_SHOTGUN_SPREAD * 16;
2001-05-06 20:50:27 +00:00
}
else
{
//Elder: fill in shotType
shotType = WP_HANDCANNON;
r = Q_crandom(&seed) * DEFAULT_SHOTGUN_HSPREAD * 16 * 4;
u = Q_crandom(&seed) * DEFAULT_SHOTGUN_VSPREAD * 16 * hc_multipler;
// r = crandom() * DEFAULT_HANDCANNON_SPREAD * 16 * 4;
// u = crandom() * DEFAULT_HANDCANNON_SPREAD * 16 * 4;
}
VectorMA( origin, 8192 * 16, forward, end);
2001-05-06 20:50:27 +00:00
VectorMA (end, r, right, end);
VectorMA (end, u, up, end);
CG_ShotgunPellet( origin, end, otherEntNum, shotType );
2001-05-06 20:50:27 +00:00
}
//Reset shellHits once we're finished with it
cg.shellHits = 0;
2001-05-06 20:50:27 +00:00
}
/*
==============
CG_ShotgunFire
==============
*/
void CG_ShotgunFire( entityState_t *es, qboolean ism3) {
vec3_t v;
int contents;
VectorSubtract( es->origin2, es->pos.trBase, v );
VectorNormalize( v );
VectorScale( v, 32, v );
VectorAdd( es->pos.trBase, v, v );
if ( cgs.glconfig.hardwareType != GLHW_RAGEPRO ) {
// ragepro can't alpha fade, so don't even bother with smoke
vec3_t up;
contents = trap_CM_PointContents( es->pos.trBase, 0 );
if ( !( contents & CONTENTS_WATER ) ) {
VectorSet( up, 0, 0, 8 );
CG_SmokePuff( v, up, 32, 1, 1, 1, 0.33f, 900, cg.time, 0, LEF_PUFF_DONT_SCALE, cgs.media.shotgunSmokePuffShader );
}
}
//Elder: note param changes
if (ism3)
2001-05-06 20:50:27 +00:00
{
CG_ShotgunPattern( es->pos.trBase, es->origin2, es->otherEntityNum, WP_M3, es->eventParm);
2001-05-06 20:50:27 +00:00
}
else
{
CG_ShotgunPattern( es->pos.trBase, es->origin2, es->otherEntityNum, WP_HANDCANNON, es->eventParm);
es->angles2[1] += 20;
//es->origin2[1] += 5;
CG_ShotgunPattern( es->pos.trBase, es->origin2, es->otherEntityNum, -1, es->eventParm);
2001-05-06 20:50:27 +00:00
}
}
/*
============================================================================
BULLETS
============================================================================
*/
/*
===============
CG_Tracer
===============
*/
void CG_Tracer( vec3_t source, vec3_t dest ) {
vec3_t forward, right;
polyVert_t verts[4];
vec3_t line;
float len, begin, end;
vec3_t start, finish;
vec3_t midpoint;
// tracer
VectorSubtract( dest, source, forward );
len = VectorNormalize( forward );
// start at least a little ways from the muzzle
if ( len < 100 ) {
return;
}
begin = 50 + random() * (len - 60);
end = begin + cg_tracerLength.value;
if ( end > len ) {
end = len;
}
VectorMA( source, begin, forward, start );
VectorMA( source, end, forward, finish );
line[0] = DotProduct( forward, cg.refdef.viewaxis[1] );
line[1] = DotProduct( forward, cg.refdef.viewaxis[2] );
VectorScale( cg.refdef.viewaxis[1], line[1], right );
VectorMA( right, -line[0], cg.refdef.viewaxis[2], right );
VectorNormalize( right );
VectorMA( finish, cg_tracerWidth.value, right, verts[0].xyz );
verts[0].st[0] = 0;
verts[0].st[1] = 1;
verts[0].modulate[0] = 255;
verts[0].modulate[1] = 255;
verts[0].modulate[2] = 255;
verts[0].modulate[3] = 255;
VectorMA( finish, -cg_tracerWidth.value, right, verts[1].xyz );
verts[1].st[0] = 1;
verts[1].st[1] = 0;
verts[1].modulate[0] = 255;
verts[1].modulate[1] = 255;
verts[1].modulate[2] = 255;
verts[1].modulate[3] = 255;
VectorMA( start, -cg_tracerWidth.value, right, verts[2].xyz );
verts[2].st[0] = 1;
verts[2].st[1] = 1;
verts[2].modulate[0] = 255;
verts[2].modulate[1] = 255;
verts[2].modulate[2] = 255;
verts[2].modulate[3] = 255;
VectorMA( start, cg_tracerWidth.value, right, verts[3].xyz );
verts[3].st[0] = 0;
verts[3].st[1] = 0;
verts[3].modulate[0] = 255;
verts[3].modulate[1] = 255;
verts[3].modulate[2] = 255;
verts[3].modulate[3] = 255;
trap_R_AddPolyToScene( cgs.media.tracerShader, 4, verts );
midpoint[0] = ( start[0] + finish[0] ) * 0.5;
midpoint[1] = ( start[1] + finish[1] ) * 0.5;
midpoint[2] = ( start[2] + finish[2] ) * 0.5;
// add the tracer sound
trap_S_StartSound( midpoint, ENTITYNUM_WORLD, CHAN_AUTO, cgs.media.tracerSound );
}
/*
======================
CG_CalcMuzzlePoint
======================
*/
static qboolean CG_CalcMuzzlePoint( int entityNum, vec3_t muzzle ) {
vec3_t forward;
centity_t *cent;
int anim;
if ( entityNum == cg.snap->ps.clientNum ) {
VectorCopy( cg.snap->ps.origin, muzzle );
muzzle[2] += cg.snap->ps.viewheight;
AngleVectors( cg.snap->ps.viewangles, forward, NULL, NULL );
VectorMA( muzzle, 14, forward, muzzle );
return qtrue;
}
cent = &cg_entities[entityNum];
if ( !cent->currentValid ) {
return qfalse;
}
VectorCopy( cent->currentState.pos.trBase, muzzle );
AngleVectors( cent->currentState.apos.trBase, forward, NULL, NULL );
anim = cent->currentState.legsAnim & ~ANIM_TOGGLEBIT;
if ( anim == LEGS_WALKCR || anim == LEGS_IDLECR ) {
muzzle[2] += CROUCH_VIEWHEIGHT;
} else {
muzzle[2] += DEFAULT_VIEWHEIGHT;
}
VectorMA( muzzle, 14, forward, muzzle );
return qtrue;
}
/*
======================
CG_Bullet
Renders bullet effects.
Elder: added soundType conditional
2001-05-06 20:50:27 +00:00
======================
*/
void CG_Bullet( vec3_t end, int sourceEntityNum, vec3_t normal,
qboolean flesh, int fleshEntityNum, impactSound_t soundType) {
2001-05-06 20:50:27 +00:00
trace_t trace;
int sourceContentType, destContentType;
2001-12-17 15:06:41 +00:00
vec3_t start, viewAngle;
centity_t *cent;
2001-05-06 20:50:27 +00:00
// if the shooter is currently valid, calc a source point and possibly
// do trail effects
if ( sourceEntityNum >= 0 && cg_tracerChance.value > 0 ) {
if ( CG_CalcMuzzlePoint( sourceEntityNum, start ) ) {
sourceContentType = trap_CM_PointContents( start, 0 );
destContentType = trap_CM_PointContents( end, 0 );
// do a complete bubble trail if necessary
if ( ( sourceContentType == destContentType ) && ( sourceContentType & CONTENTS_WATER ) ) {
CG_BubbleTrail( start, end, 32 );
}
// bubble trail from water into air
else if ( ( sourceContentType & CONTENTS_WATER ) ) {
trap_CM_BoxTrace( &trace, end, start, NULL, NULL, 0, CONTENTS_WATER );
CG_BubbleTrail( start, trace.endpos, 32 );
}
// bubble trail from air into water
else if ( ( destContentType & CONTENTS_WATER ) ) {
trap_CM_BoxTrace( &trace, start, end, NULL, NULL, 0, CONTENTS_WATER );
CG_BubbleTrail( trace.endpos, end, 32 );
// TODO: water splash mark
2001-05-06 20:50:27 +00:00
}
// draw a tracer
// Elder: only if not using SSG, check if this client is the source
if (sourceEntityNum == cg.snap->ps.clientNum)
{
if (cg.snap->ps.weapon != WP_SSG3000)
{
if ( random() < cg_tracerChance.value )
CG_Tracer( start, end );
}
}
else
{
cent = &cg_entities[sourceEntityNum];
if ( cent->currentValid && cent->currentState.weapon != WP_SSG3000)
{
if ( random() < cg_tracerChance.value ) {
CG_Tracer( start, end );
}
}
2001-05-06 20:50:27 +00:00
}
}
}
2001-12-17 15:06:41 +00:00
// NiceAss: Added for better sparking.
CG_CalcViewDir(sourceEntityNum, end, viewAngle);
2001-05-06 20:50:27 +00:00
// impact splash and mark
if ( flesh ) {
CG_Bleed( end, fleshEntityNum );
2001-05-06 20:50:27 +00:00
} else {
//Blaze: Changed WP_MACHINEGUN to WP_PISTOL
2001-12-17 15:06:41 +00:00
CG_MissileHitWall( WP_PISTOL, 0, end, normal, viewAngle, soundType, 0 );
2001-05-06 20:50:27 +00:00
}
}
/*
==================
CG_LocalLaser
Elder:
Local laser dot if it is the client's own laser
==================
*/
static void CG_LocalLaser ()
{
vec3_t muzzle;
vec3_t forward;
vec3_t end;
refEntity_t *re;
trace_t tr;
//Create the laser entity if it's not there
if (cg.laserSight == qfalse)
{
//CG_Printf("Initializing Local Laser...\n");
cg.laserSight = qtrue;
cg.laserEnt = CG_AllocLocalEntity();
cg.laserEnt->startTime = cg.time;
cg.laserEnt->color[3] = 1.0;
//cg.laserEnt->pos.trType = TR_INTERPOLATE;
}
//Setup refEntity stuff
re = &cg.laserEnt->refEntity;
re->radius = 6;
re->reType = RT_SPRITE;
re->rotation = 0;
re->customShader = cgs.media.laserShader;
//Calculate muzzle and endpoint
if (CG_CalcMuzzlePoint(cg.snap->ps.clientNum, muzzle))
{
AngleVectors( cg.snap->ps.viewangles, forward, NULL, NULL );
VectorMA( muzzle, 8192 * 16, forward, end );
}
else
{
CG_Error("CG_LocalLaser: Could not calculate own muzzle point\n");
}
CG_Trace(&tr, muzzle, NULL, NULL, end, cg.predictedPlayerState.clientNum, CONTENTS_SOLID);
//Set position of laser dot
if (tr.fraction != 1)
VectorMA(tr.endpos,-4, forward, tr.endpos);
VectorCopy(re->origin, re->oldorigin);
VectorCopy(tr.endpos, re->origin);
//VectorCopy(tr.endpos, cg.laserEnt->pos.trBase);
//CG_EvaluateTrajectory(&cg.laserEnt->pos, cg.time, re->origin);
//Boost the endTime
cg.laserEnt->endTime += 10000;
//if (tr.surfaceFlags & SURF_NOIMPACT || tr.surfaceFlags & SURF_SKY)
//Don't render if it hits the sky
//if (!(tr.surfaceFlags & SURF_SKY))
trap_R_AddRefEntityToScene( re );
}
/*
==================
CG_CheckLaser
Elder:
Whether or not to use the local laser
Broken until I find a way to lerp an entity without a cent
==================
*/
void CG_CheckLaser()
{
//Elder: check for local laser
if (bg_itemlist[cg.snap->ps.stats[STAT_HOLDABLE_ITEM]].giTag == HI_LASER &&
cg_RQ3_laserAssist.integer &&
(cg.snap->ps.weapon == WP_PISTOL ||
cg.snap->ps.weapon == WP_MP5 ||
cg.snap->ps.weapon == WP_M4))
{
CG_LocalLaser();
}
//Disable laser
else if (cg.laserSight == qtrue)
{
//CG_Printf("Destroying Local Laser...\n");
CG_FreeLocalEntity(cg.laserEnt);
cg.laserSight = qfalse;
}
}
/*
==================
CG_ReloadWeapon
Elder: Does local reload effects like
reload shell drops, sounds, etc.
==================
*/
void CG_ReloadWeapon (centity_t *cent, int reloadStage)
{
entityState_t *ent;
weaponInfo_t *weap;
ent = &cent->currentState;
2001-11-11 07:23:16 +00:00
if ( ent->weapon == WP_NONE )
return;
2001-11-11 07:23:16 +00:00
if ( ent->weapon >= WP_NUM_WEAPONS )
{
CG_Error( "CG_ReloadWeapon: ent->weapon >= WP_NUM_WEAPONS" );
return;
}
2001-11-11 07:23:16 +00:00
weap = &cg_weapons[ent->weapon];
switch ( reloadStage )
{
2001-11-11 07:23:16 +00:00
case 0:
if (weap->worldReloadSound[0] && ent->clientNum != cg.snap->ps.clientNum)
{
2001-11-11 07:23:16 +00:00
trap_S_StartSound(cent->lerpOrigin, ent->number, CHAN_AUTO, weap->worldReloadSound[0]);
}
2001-11-11 07:23:16 +00:00
break;
case 1:
if (weap->worldReloadSound[1] && ent->clientNum != cg.snap->ps.clientNum)
{
2001-11-11 07:23:16 +00:00
trap_S_StartSound(cent->lerpOrigin, ent->number, CHAN_AUTO, weap->worldReloadSound[1]);
}
2001-11-11 07:23:16 +00:00
if ( ent->weapon == WP_HANDCANNON )
CG_ShotgunEjectBrass( cent );
break;
case 2:
if (weap->worldReloadSound[2] && ent->clientNum != cg.snap->ps.clientNum)
{
2001-11-11 07:23:16 +00:00
trap_S_StartSound(cent->lerpOrigin, ent->number, CHAN_AUTO, weap->worldReloadSound[2]);
}
2001-11-25 23:17:41 +00:00
2001-11-11 07:23:16 +00:00
break;
default:
CG_Error("CG_ReloadWeapon: Reload stage > 2\n");
break;
}
}
2001-11-11 07:23:16 +00:00
/*
=================
CG_CalcViewAngle
Added by NiceAss.
Start not known. End known.
Used for calculating a player's viewing angle.
Currently used for spark directions (reflected off a plane).
=================
*/
void CG_CalcViewDir(const int sourceEntityNum, const vec3_t end, vec3_t viewDir) {
vec3_t delta, start;
CG_CalcMuzzlePoint(sourceEntityNum, start);
VectorSubtract(end, start, delta);
VectorNormalize2(delta, viewDir);
}
/*
=================
CG_CalcViewAngle2
Added by NiceAss.
Start known. End known.
Used for calculating a player's viewing angle.
Currently used for spark directions (reflected off a plane).
=================
*/
void CG_CalcViewDir2(const vec3_t start, const vec3_t end, vec3_t viewDir) {
vec3_t delta;
VectorSubtract(end, start, delta);
VectorNormalize2(delta, viewDir);
}