/* =========================================================================== Copyright (C) 1999 - 2005, Id Software, Inc. Copyright (C) 2000 - 2013, Raven Software, Inc. Copyright (C) 2001 - 2013, Activision, Inc. Copyright (C) 2013 - 2015, OpenJK contributors This file is part of the OpenJK source code. OpenJK is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License version 2 as published by the Free Software Foundation. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, see . =========================================================================== */ #include "common_headers.h" #include "../rd-common/tr_public.h" // bg_pmove.c -- both games player movement code // takes a playerstate and a usercmd as input and returns a modifed playerstate // define GAME_INCLUDE so that g_public.h does not define the // short, server-visible gclient_t and gentity_t structures, // because we define the full size ones in this file #define GAME_INCLUDE #include "../qcommon/q_shared.h" #include "g_shared.h" #include "bg_local.h" #include "g_local.h" #include "g_functions.h" #include "anims.h" #include "../cgame/cg_local.h" // yeah I know this is naughty, but we're shipping soon... #include "wp_saber.h" #include "g_vehicles.h" #include extern qboolean G_DoDismemberment( gentity_t *self, vec3_t point, int mod, int damage, int hitLoc, qboolean force = qfalse ); extern qboolean G_EntIsUnlockedDoor( int entityNum ); extern qboolean G_EntIsDoor( int entityNum ); extern qboolean InFront( vec3_t spot, vec3_t from, vec3_t fromAngles, float threshHold = 0.0f ); extern void G_AddVoiceEvent( gentity_t *self, int event, int speakDebounceTime ); extern qboolean Q3_TaskIDPending( gentity_t *ent, taskID_t taskType ); extern qboolean WP_SaberLose( gentity_t *self, vec3_t throwDir ); extern int Jedi_ReCalcParryTime( gentity_t *self, evasionType_t evasionType ); extern qboolean PM_HasAnimation( gentity_t *ent, int animation ); extern saberMoveName_t PM_SaberAnimTransitionMove( saberMoveName_t curmove, saberMoveName_t newmove ); extern saberMoveName_t PM_AttackMoveForQuad( int quad ); extern qboolean PM_SaberInTransition( int move ); extern qboolean PM_SaberInTransitionAny( int move ); extern qboolean PM_SaberInBounce( int move ); extern qboolean PM_SaberInSpecialAttack( int anim ); extern qboolean PM_SaberInAttack( int move ); extern qboolean PM_InAnimForSaberMove( int anim, int saberMove ); extern saberMoveName_t PM_SaberBounceForAttack( int move ); extern saberMoveName_t PM_SaberAttackForMovement( int forwardmove, int rightmove, int curmove ); extern saberMoveName_t PM_BrokenParryForParry( int move ); extern saberMoveName_t PM_KnockawayForParry( int move ); extern qboolean PM_SaberInParry( int move ); extern qboolean PM_SaberInKnockaway( int move ); extern qboolean PM_SaberInBrokenParry( int move ); extern qboolean PM_SaberInReflect( int move ); extern qboolean PM_SaberInIdle( int move ); extern qboolean PM_SaberInStart( int move ); extern qboolean PM_SaberInReturn( int move ); extern qboolean PM_SaberKataDone( int curmove, int newmove ); extern qboolean PM_SaberInSpecial( int move ); extern qboolean PM_InDeathAnim ( void ); extern qboolean PM_StandingAnim( int anim ); extern qboolean PM_KickMove( int move ); extern qboolean PM_KickingAnim( int anim ); extern qboolean PM_InAirKickingAnim( int anim ); extern qboolean PM_SuperBreakLoseAnim( int anim ); extern qboolean PM_SuperBreakWinAnim( int anim ); extern qboolean PM_InCartwheel( int anim ); extern qboolean PM_InButterfly( int anim ); extern qboolean PM_CanRollFromSoulCal( playerState_t *ps ); extern saberMoveName_t PM_SaberFlipOverAttackMove( void ); extern qboolean PM_CheckFlipOverAttackMove( qboolean checkEnemy ); extern saberMoveName_t PM_SaberJumpForwardAttackMove( void ); extern qboolean PM_CheckJumpForwardAttackMove( void ); extern saberMoveName_t PM_SaberBackflipAttackMove( void ); extern qboolean PM_CheckBackflipAttackMove( void ); extern saberMoveName_t PM_SaberDualJumpAttackMove( void ); extern qboolean PM_CheckDualJumpAttackMove( void ); extern saberMoveName_t PM_SaberLungeAttackMove( qboolean fallbackToNormalLunge ); extern qboolean PM_CheckLungeAttackMove( void ); extern qboolean PM_InSecondaryStyle( void ); extern qboolean PM_KnockDownAnimExtended( int anim ); extern void G_StartMatrixEffect( gentity_t *ent, int meFlags = 0, int length = 1000, float timeScale = 0.0f, int spinTime = 0 ); extern void WP_ForcePowerStop( gentity_t *self, forcePowers_t forcePower ); extern qboolean WP_ForcePowerAvailable( gentity_t *self, forcePowers_t forcePower, int overrideAmt ); extern void WP_ForcePowerDrain( gentity_t *self, forcePowers_t forcePower, int overrideAmt ); extern float G_ForceWallJumpStrength( void ); extern qboolean G_CheckRollSafety( gentity_t *self, int anim, float testDist ); extern saberMoveName_t PM_CheckDualSpinProtect( void ); extern saberMoveName_t PM_CheckPullAttack( void ); extern qboolean JET_Flying( gentity_t *self ); extern void JET_FlyStart( gentity_t *self ); extern void JET_FlyStop( gentity_t *self ); extern qboolean PM_LockedAnim( int anim ); extern qboolean G_TryingKataAttack( gentity_t *self, usercmd_t *cmd ); extern qboolean G_TryingCartwheel( gentity_t *self, usercmd_t *cmd ); extern qboolean G_TryingSpecial( gentity_t *self, usercmd_t *cmd ); extern qboolean G_TryingJumpAttack( gentity_t *self, usercmd_t *cmd ); extern qboolean G_TryingJumpForwardAttack( gentity_t *self, usercmd_t *cmd ); extern void WP_SaberSwingSound( gentity_t *ent, int saberNum, swingType_t swingType ); extern qboolean WP_UseFirstValidSaberStyle( gentity_t *ent, int *saberAnimLevel ); extern qboolean WP_SaberStyleValidForSaber( gentity_t *ent, int saberAnimLevel ); qboolean PM_InKnockDown( playerState_t *ps ); qboolean PM_InKnockDownOnGround( playerState_t *ps ); qboolean PM_InGetUp( playerState_t *ps ); qboolean PM_InRoll( playerState_t *ps ); qboolean PM_SpinningSaberAnim( int anim ); qboolean PM_GettingUpFromKnockDown( float standheight, float crouchheight ); qboolean PM_SpinningAnim( int anim ); qboolean PM_FlippingAnim( int anim ); qboolean PM_PainAnim( int anim ); qboolean PM_RollingAnim( int anim ); qboolean PM_SwimmingAnim( int anim ); qboolean PM_InReboundJump( int anim ); qboolean PM_ForceJumpingAnim( int anim ); void PM_CmdForRoll( playerState_t *ps, usercmd_t *pCmd ); extern int parryDebounce[]; extern qboolean cg_usingInFrontOf; extern qboolean player_locked; extern qboolean MatrixMode; qboolean waterForceJump; extern cvar_t *g_timescale; extern cvar_t *g_speederControlScheme; extern cvar_t *d_slowmodeath; extern cvar_t *g_debugMelee; extern cvar_t *g_saberNewControlScheme; extern cvar_t *g_stepSlideFix; extern cvar_t *g_saberAutoBlocking; static void PM_SetWaterLevelAtPoint( vec3_t org, int *waterlevel, int *watertype ); #define FLY_NONE 0 #define FLY_NORMAL 1 #define FLY_VEHICLE 2 #define FLY_HOVER 3 int Flying = FLY_NONE; pmove_t *pm; pml_t pml; // movement parameters const float pm_stopspeed = 100.0f; const float pm_duckScale = 0.50f; const float pm_swimScale = 0.50f; float pm_ladderScale = 0.7f; const float pm_vehicleaccelerate = 36.0f; const float pm_accelerate = 12.0f; const float pm_airaccelerate = 4.0f; const float pm_wateraccelerate = 4.0f; const float pm_flyaccelerate = 8.0f; const float pm_friction = 6.0f; const float pm_waterfriction = 1.0f; const float pm_flightfriction = 3.0f; //const float pm_frictionModifier = 3.0f; //Used for "careful" mode (when pressing use) const float pm_airDecelRate = 1.35f; //Used for air decelleration away from current movement velocity int c_pmove = 0; extern void PM_SetTorsoAnimTimer( gentity_t *ent, int *torsoAnimTimer, int time ); extern void PM_SetLegsAnimTimer( gentity_t *ent, int *legsAnimTimer, int time ); extern void PM_TorsoAnimation( void ); extern int PM_TorsoAnimForFrame( gentity_t *ent, int torsoFrame ); extern int PM_AnimLength( int index, animNumber_t anim ); extern qboolean PM_InDeathAnim ( void ); extern qboolean PM_InOnGroundAnim ( playerState_t *ps ); extern weaponInfo_t cg_weapons[MAX_WEAPONS]; extern int PM_PickAnim( gentity_t *self, int minAnim, int maxAnim ); extern void DoImpact( gentity_t *self, gentity_t *other, qboolean damageSelf, trace_t *trace ); #define PHASER_RECHARGE_TIME 100 extern saberMoveName_t transitionMove[Q_NUM_QUADS][Q_NUM_QUADS]; extern Vehicle_t *G_IsRidingVehicle( gentity_t *ent ); Vehicle_t *PM_RidingVehicle( void ) { return (G_IsRidingVehicle( pm->gent )); } extern qboolean G_ControlledByPlayer( gentity_t *self ); qboolean PM_ControlledByPlayer( void ) { return G_ControlledByPlayer( pm->gent ); } qboolean BG_UnrestrainedPitchRoll( playerState_t *ps, Vehicle_t *pVeh ) { /* if ( 0 && ps->clientNum < MAX_CLIENTS //real client && ps->m_iVehicleNum//in a vehicle && pVeh //valid vehicle data pointer && pVeh->m_pVehicleInfo//valid vehicle info && pVeh->m_pVehicleInfo->type == VH_FIGHTER )//fighter //FIXME: specify per vehicle instead of assuming true for all fighters //FIXME: map/server setting? {//can roll and pitch without limitation! return qtrue; }*/ return qfalse; } /* =============== PM_AddEvent =============== */ void PM_AddEvent( int newEvent ) { AddEventToPlayerstate( newEvent, 0, pm->ps ); } qboolean PM_PredictJumpSafe( vec3_t jumpHorizDir, float jumpHorizSpeed, float jumpVertSpeed, int predictTimeLength ) { return qtrue; } void PM_GrabWallForJump( int anim ) {//NOTE!!! assumes an appropriate anim is being passed in!!! PM_SetAnim( pm, SETANIM_BOTH, anim, SETANIM_FLAG_RESTART|SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD, 0 ); PM_AddEvent( EV_JUMP );//make sound for grab pm->ps->pm_flags |= PMF_STUCK_TO_WALL; } qboolean PM_CheckGrabWall( trace_t *trace ) { if ( !pm->gent || !pm->gent->client ) { return qfalse; } if ( pm->gent->health <= 0 ) {//must be alive return qfalse; } if ( pm->gent->client->ps.groundEntityNum != ENTITYNUM_NONE ) {//must be in air return qfalse; } if ( trace->plane.normal[2] != 0 ) {//must be a flat wall return qfalse; } if ( !trace->plane.normal[0] && !trace->plane.normal[1] ) {//invalid normal return qfalse; } if ( (trace->contents&(CONTENTS_PLAYERCLIP|CONTENTS_MONSTERCLIP)) ) {//can't jump off of clip brushes return qfalse; } if ( pm->gent->client->ps.forcePowerLevel[FP_LEVITATION] < FORCE_LEVEL_1 ) {//must have at least FJ 1 return qfalse; } if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && pm->gent->client->ps.forcePowerLevel[FP_LEVITATION] < FORCE_LEVEL_3 ) {//player must have force jump 3 return qfalse; } if ( (pm->ps->saber[0].saberFlags&SFL_NO_WALL_GRAB) ) { return qfalse; } if ( pm->ps->dualSabers && (pm->ps->saber[1].saberFlags&SFL_NO_WALL_GRAB) ) { return qfalse; } if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) ) {//player //only if we were in a longjump if ( pm->ps->legsAnim != BOTH_FORCELONGLEAP_START && pm->ps->legsAnim != BOTH_FORCELONGLEAP_ATTACK ) { return qfalse; } //hit a flat wall during our long jump, see if we should grab it vec3_t moveDir; VectorCopy( pm->ps->velocity, moveDir ); VectorNormalize( moveDir ); if ( DotProduct( moveDir, trace->plane.normal ) > -0.65f ) {//not enough of a direct impact, just slide off return qfalse; } if ( fabs(trace->plane.normal[2]) > MAX_WALL_GRAB_SLOPE ) { return qfalse; } //grab it! //FIXME: stop Matrix effect! VectorClear( pm->ps->velocity ); //FIXME: stop slidemove! //NOTE: we know it's forward, so... PM_GrabWallForJump( BOTH_FORCEWALLREBOUND_FORWARD ); return qtrue; } else {//NPCs if ( PM_InReboundJump( pm->ps->legsAnim ) ) {//already in a rebound! return qfalse; } if ( (pm->ps->eFlags&EF_FORCE_GRIPPED) ) {//being gripped! return qfalse; } /* if ( pm->gent->painDebounceTime > level.time ) {//can't move! return qfalse; } if ( (pm->ps->pm_flags&PMF_TIME_KNOCKBACK) ) {//being thrown back! return qfalse; } */ if ( pm->gent->NPC && (pm->gent->NPC->aiFlags&NPCAI_DIE_ON_IMPACT) ) {//faling to our death! return qfalse; } //FIXME: random chance, based on skill/rank? if ( pm->ps->legsAnim != BOTH_FORCELONGLEAP_START && pm->ps->legsAnim != BOTH_FORCELONGLEAP_ATTACK ) {//not in a long-jump if ( !pm->gent->enemy ) {//no enemy return qfalse; } else {//see if the enemy is in the direction of the wall or above us //if ( pm->gent->enemy->currentOrigin[2] < (pm->ps->origin[2]-128) ) {//enemy is way below us vec3_t enemyDir; VectorSubtract( pm->gent->enemy->currentOrigin, pm->ps->origin, enemyDir ); enemyDir[2] = 0; VectorNormalize( enemyDir ); if ( DotProduct( enemyDir, trace->plane.normal ) < 0.65f ) {//jumping off this wall would not launch me in the general direction of my enemy return qfalse; } } } } //FIXME: check for ground close beneath us? //FIXME: check for obstructions in the dir we're going to jump // - including "do not enter" brushes! //hit a flat wall during our long jump, see if we should grab it vec3_t moveDir; VectorCopy( pm->ps->velocity, moveDir ); VectorNormalize( moveDir ); if ( DotProduct( moveDir, trace->plane.normal ) > -0.65f ) {//not enough of a direct impact, just slide off return qfalse; } //Okay, now see if jumping off this thing would send us into a do not enter brush if ( !PM_PredictJumpSafe( trace->plane.normal, JUMP_OFF_WALL_SPEED, G_ForceWallJumpStrength(), 1500 ) ) {//we would hit a do not enter brush, so don't grab the wall return qfalse; } //grab it! //Pick the proper anim int anim = BOTH_FORCEWALLREBOUND_FORWARD; vec3_t facingAngles, wallDir, fwdDir, rtDir; VectorSubtract( trace->endpos, pm->gent->lastOrigin, wallDir ); wallDir[2] = 0; VectorNormalize( wallDir ); VectorSet( facingAngles, 0, pm->ps->viewangles[YAW], 0 ); AngleVectors( facingAngles, fwdDir, rtDir, NULL ); float fDot = DotProduct( fwdDir, wallDir ); if ( fabs( fDot ) >= 0.5f ) {//hit a wall in front/behind if ( fDot > 0.0f ) {//in front anim = BOTH_FORCEWALLREBOUND_FORWARD; } else {//behind anim = BOTH_FORCEWALLREBOUND_BACK; } } else if ( DotProduct( rtDir, wallDir ) > 0 ) {//hit a wall on the right anim = BOTH_FORCEWALLREBOUND_RIGHT; } else {//hit a wall on the left anim = BOTH_FORCEWALLREBOUND_LEFT; } VectorClear( pm->ps->velocity ); //FIXME: stop slidemove! PM_GrabWallForJump( anim ); return qtrue; } //return qfalse; } /* =============== qboolean PM_ClientImpact( trace_t *trace, qboolean damageSelf ) =============== */ qboolean PM_ClientImpact( trace_t *trace, qboolean damageSelf ) { gentity_t *traceEnt; int otherEntityNum = trace->entityNum; if ( !pm->gent ) { return qfalse; } traceEnt = &g_entities[otherEntityNum]; if ( otherEntityNum == ENTITYNUM_WORLD || (traceEnt->bmodel && traceEnt->s.pos.trType == TR_STATIONARY ) ) {//hit world or a non-moving brush if ( PM_CheckGrabWall( trace ) ) {//stopped on the wall return qtrue; } } if( (VectorLength( pm->ps->velocity )*(pm->gent->mass/10)) >= 100 && (pm->gent->client->NPC_class == CLASS_VEHICLE || pm->ps->lastOnGround+100material>=MAT_GLASS&&pm->gent->lastImpact+100<=level.time)) { DoImpact( pm->gent, &g_entities[otherEntityNum], damageSelf, trace ); } if ( otherEntityNum >= ENTITYNUM_WORLD ) { return qfalse; } if ( !traceEnt || !(traceEnt->contents&pm->tracemask) ) {//it's dead or not in my way anymore return qtrue; } return qfalse; } /* =============== PM_AddTouchEnt =============== */ void PM_AddTouchEnt( int entityNum ) { int i; if ( entityNum == ENTITYNUM_WORLD ) { return; } if ( pm->numtouch == MAXTOUCH ) { return; } // see if it is already added for ( i = 0 ; i < pm->numtouch ; i++ ) { if ( pm->touchents[ i ] == entityNum ) { return; } } // add it pm->touchents[pm->numtouch] = entityNum; pm->numtouch++; } /* ================== PM_ClipVelocity Slide off of the impacting surface This will pull you down onto slopes if heading away from them and push you up them as you go up them. Also stops you when you hit walls. ================== */ void PM_ClipVelocity( vec3_t in, vec3_t normal, vec3_t out, float overbounce ) { float backoff; float change; float oldInZ; int i; if ( (pm->ps->pm_flags&PMF_STUCK_TO_WALL) ) {//no sliding! VectorCopy( in, out ); return; } oldInZ = in[2]; backoff = DotProduct (in, normal); if ( backoff < 0 ) { backoff *= overbounce; } else { backoff /= overbounce; } for ( i=0 ; i<3 ; i++ ) { change = normal[i]*backoff; /* if ( i == 2 && Flying == FLY_HOVER && change > 0 ) {//don't pull a hovercraft down change = 0; } else */ { out[i] = in[i] - change; } } if ( g_stepSlideFix->integer ) { if ( pm->ps->clientNum < MAX_CLIENTS//normal player && normal[2] < MIN_WALK_NORMAL )//sliding against a steep slope { if ( pm->ps->groundEntityNum != ENTITYNUM_NONE )//on the ground {//if walking on the ground, don't slide up slopes that are too steep to walk on out[2] = oldInZ; } } } } /* ================== PM_Friction Handles both ground friction and water friction ================== */ static void PM_Friction( void ) { vec3_t vec; float *vel; float speed, newspeed, control; float drop, friction = pm->ps->friction; vel = pm->ps->velocity; VectorCopy( vel, vec ); if ( pml.walking ) { vec[2] = 0; // ignore slope movement } speed = VectorLength(vec); if (speed < 1) { vel[0] = 0; vel[1] = 0; // allow sinking underwater // FIXME: still have z friction underwater? return; } drop = 0; // apply ground friction, even if on ladder if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_VEHICLE && pm->gent->m_pVehicle && pm->gent->m_pVehicle->m_pVehicleInfo->type != VH_ANIMAL ) { friction = pm->gent->m_pVehicle->m_pVehicleInfo->friction; if ( pm->gent->m_pVehicle && pm->gent->m_pVehicle->m_pVehicleInfo->hoverHeight > 0 ) {//in a hovering vehicle, have air control if ( pm->gent->m_pVehicle->m_ulFlags & VEH_FLYING ) { friction = 0.10f; } } if ( !(pm->ps->pm_flags & PMF_TIME_KNOCKBACK) && !(pm->ps->pm_flags & PMF_TIME_NOFRICTION) ) { control = speed < pm_stopspeed ? pm_stopspeed : speed; drop += control*friction*pml.frametime; /* if ( Flying == FLY_HOVER ) { if ( pm->cmd.rightmove ) {//if turning, increase friction control *= 2.0f; } if ( pm->ps->groundEntityNum < ENTITYNUM_NONE ) {//on the ground drop += control*friction*pml.frametime; } else if ( pml.groundPlane ) {//on a slope drop += control*friction*2.0f*pml.frametime; } else {//in air drop += control*2.0f*friction*pml.frametime; } } */ } } else if ( Flying != FLY_NORMAL ) { if ( (pm->watertype & CONTENTS_LADDER) || pm->waterlevel <= 1 ) { if ( (pm->watertype & CONTENTS_LADDER) || (pml.walking && !(pml.groundTrace.surfaceFlags & SURF_SLICK)) ) { // if getting knocked back, no friction if ( !(pm->ps->pm_flags & PMF_TIME_KNOCKBACK) && !(pm->ps->pm_flags & PMF_TIME_NOFRICTION) ) { if ( pm->ps->legsAnim == BOTH_FORCELONGLEAP_START || pm->ps->legsAnim == BOTH_FORCELONGLEAP_ATTACK || pm->ps->legsAnim == BOTH_FORCELONGLEAP_LAND ) {//super forward jump if ( pm->ps->groundEntityNum != ENTITYNUM_NONE ) {//not in air if ( pm->cmd.forwardmove < 0 ) {//trying to hold back some friction *= 0.5f;//0.25f; } else {//free slide friction *= 0.2f;//0.1f; } pm->cmd.forwardmove = pm->cmd.rightmove = 0; if ( pml.groundPlane && pm->ps->legsAnim == BOTH_FORCELONGLEAP_LAND ) { //slide effect G_PlayEffect( "env/slide_dust", pml.groundTrace.endpos, pml.groundTrace.plane.normal ); //FIXME: slide sound } } } /* else if ( pm->cmd.buttons & BUTTON_USE ) {//If the use key is pressed. slow the player more quickly if ( pm->gent->client->NPC_class != CLASS_VEHICLE ) // if not in a vehicle... {//in a vehicle, use key makes you turbo-boost friction *= pm_frictionModifier; } } */ control = speed < pm_stopspeed ? pm_stopspeed : speed; drop += control*friction*pml.frametime; } } } } else if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && pm->gent && pm->gent->client && (pm->gent->client->NPC_class == CLASS_BOBAFETT || pm->gent->client->NPC_class == CLASS_ROCKETTROOPER) && pm->gent->client->moveType == MT_FLYSWIM ) {//player as Boba drop += speed*pm_waterfriction*pml.frametime; } if ( Flying == FLY_VEHICLE ) { if ( !(pm->ps->pm_flags & PMF_TIME_KNOCKBACK) && !(pm->ps->pm_flags & PMF_TIME_NOFRICTION) ) { control = speed < pm_stopspeed ? pm_stopspeed : speed; drop += control*friction*pml.frametime; } } // apply water friction even if just wading if ( !waterForceJump ) { if ( pm->waterlevel && !(pm->watertype & CONTENTS_LADDER)) { drop += speed*pm_waterfriction*pm->waterlevel*pml.frametime; } } // apply flying friction if ( pm->ps->pm_type == PM_SPECTATOR ) { drop += speed*pm_flightfriction*pml.frametime; } // scale the velocity newspeed = speed - drop; if (newspeed < 0) newspeed = 0; newspeed /= speed; vel[0] = vel[0] * newspeed; vel[1] = vel[1] * newspeed; vel[2] = vel[2] * newspeed; } /* ============== PM_Accelerate Handles user intended acceleration ============== */ static void PM_Accelerate( vec3_t wishdir, float wishspeed, float accel ) { int i; float addspeed, accelspeed, currentspeed; currentspeed = DotProduct (pm->ps->velocity, wishdir); addspeed = wishspeed - currentspeed; if (addspeed <= 0) { return; } accelspeed = ( accel * pml.frametime ) * wishspeed; if (accelspeed > addspeed) { accelspeed = addspeed; } for (i=0 ; i<3 ; i++) { pm->ps->velocity[i] += accelspeed * wishdir[i]; } } /* ============ PM_CmdScale Returns the scale factor to apply to cmd movements This allows the clients to use axial -127 to 127 values for all directions without getting a sqrt(2) distortion in speed. ============ */ static float PM_CmdScale( usercmd_t *cmd ) { int max; float total; float scale; max = abs( cmd->forwardmove ); if ( abs( cmd->rightmove ) > max ) { max = abs( cmd->rightmove ); } if ( abs( cmd->upmove ) > max ) { max = abs( cmd->upmove ); } if ( !max ) { return 0; } total = sqrt( (float)(( cmd->forwardmove * cmd->forwardmove ) + ( cmd->rightmove * cmd->rightmove ) + ( cmd->upmove * cmd->upmove )) ); scale = (float) pm->ps->speed * max / ( 127.0f * total ); return scale; } /* ================ PM_SetMovementDir Determine the rotation of the legs reletive to the facing dir ================ */ static void PM_SetMovementDir( void ) { if ( pm->cmd.forwardmove || pm->cmd.rightmove ) { if ( pm->cmd.rightmove == 0 && pm->cmd.forwardmove > 0 ) { pm->ps->movementDir = 0; } else if ( pm->cmd.rightmove < 0 && pm->cmd.forwardmove > 0 ) { pm->ps->movementDir = 1; } else if ( pm->cmd.rightmove < 0 && pm->cmd.forwardmove == 0 ) { pm->ps->movementDir = 2; } else if ( pm->cmd.rightmove < 0 && pm->cmd.forwardmove < 0 ) { pm->ps->movementDir = 3; } else if ( pm->cmd.rightmove == 0 && pm->cmd.forwardmove < 0 ) { pm->ps->movementDir = 4; } else if ( pm->cmd.rightmove > 0 && pm->cmd.forwardmove < 0 ) { pm->ps->movementDir = 5; } else if ( pm->cmd.rightmove > 0 && pm->cmd.forwardmove == 0 ) { pm->ps->movementDir = 6; } else if ( pm->cmd.rightmove > 0 && pm->cmd.forwardmove > 0 ) { pm->ps->movementDir = 7; } } else { // if they aren't actively going directly sideways, // change the animation to the diagonal so they // don't stop too crooked if ( pm->ps->movementDir == 2 ) { pm->ps->movementDir = 1; } else if ( pm->ps->movementDir == 6 ) { pm->ps->movementDir = 7; } } } /* ============= PM_CheckJump ============= */ #define METROID_JUMP 1 qboolean PM_InReboundJump( int anim ) { switch ( anim ) { case BOTH_FORCEWALLREBOUND_FORWARD: case BOTH_FORCEWALLREBOUND_LEFT: case BOTH_FORCEWALLREBOUND_BACK: case BOTH_FORCEWALLREBOUND_RIGHT: return qtrue; break; } return qfalse; } qboolean PM_InReboundHold( int anim ) { switch ( anim ) { case BOTH_FORCEWALLHOLD_FORWARD: case BOTH_FORCEWALLHOLD_LEFT: case BOTH_FORCEWALLHOLD_BACK: case BOTH_FORCEWALLHOLD_RIGHT: return qtrue; break; } return qfalse; } qboolean PM_InReboundRelease( int anim ) { switch ( anim ) { case BOTH_FORCEWALLRELEASE_FORWARD: case BOTH_FORCEWALLRELEASE_LEFT: case BOTH_FORCEWALLRELEASE_BACK: case BOTH_FORCEWALLRELEASE_RIGHT: return qtrue; break; } return qfalse; } qboolean PM_InBackFlip( int anim ) { switch ( anim ) { case BOTH_FLIP_BACK1: case BOTH_FLIP_BACK2: case BOTH_FLIP_BACK3: case BOTH_ALORA_FLIP_B: return qtrue; break; } return qfalse; } qboolean PM_InSpecialJump( int anim ) { switch ( anim ) { case BOTH_WALL_RUN_RIGHT: case BOTH_WALL_RUN_RIGHT_STOP: case BOTH_WALL_RUN_RIGHT_FLIP: case BOTH_WALL_RUN_LEFT: case BOTH_WALL_RUN_LEFT_STOP: case BOTH_WALL_RUN_LEFT_FLIP: case BOTH_WALL_FLIP_RIGHT: case BOTH_WALL_FLIP_LEFT: case BOTH_WALL_FLIP_BACK1: case BOTH_BUTTERFLY_LEFT: case BOTH_BUTTERFLY_RIGHT: case BOTH_BUTTERFLY_FL1: case BOTH_BUTTERFLY_FR1: case BOTH_FJSS_TR_BL: case BOTH_FJSS_TL_BR: case BOTH_FORCELEAP2_T__B_: case BOTH_JUMPFLIPSLASHDOWN1://# case BOTH_JUMPFLIPSTABDOWN://# case BOTH_JUMPATTACK6: case BOTH_JUMPATTACK7: case BOTH_ARIAL_LEFT: case BOTH_ARIAL_RIGHT: case BOTH_ARIAL_F1: case BOTH_CARTWHEEL_LEFT: case BOTH_CARTWHEEL_RIGHT: case BOTH_FORCELONGLEAP_START: case BOTH_FORCELONGLEAP_ATTACK: case BOTH_FORCEWALLRUNFLIP_START: case BOTH_FORCEWALLRUNFLIP_END: case BOTH_FORCEWALLRUNFLIP_ALT: case BOTH_FLIP_ATTACK7: case BOTH_FLIP_HOLD7: case BOTH_FLIP_LAND: case BOTH_A7_SOULCAL: return qtrue; } if ( PM_InReboundJump( anim ) ) { return qtrue; } if ( PM_InReboundHold( anim ) ) { return qtrue; } if ( PM_InReboundRelease( anim ) ) { return qtrue; } if ( PM_InBackFlip( anim ) ) { return qtrue; } return qfalse; } extern void CG_PlayerLockedWeaponSpeech( int jumping ); qboolean PM_ForceJumpingUp( gentity_t *gent ) { if ( !gent || !gent->client ) { return qfalse; } if ( gent->NPC ) {//this is ONLY for the player if ( player && player->client && player->client->ps.viewEntity == gent->s.number ) {//okay to jump if an NPC controlled by the player } else { return qfalse; } } if ( !(gent->client->ps.forcePowersActive&(1<client->ps.forceJumpCharge ) {//already jumped and let go return qfalse; } if ( PM_InSpecialJump( gent->client->ps.legsAnim ) ) { return qfalse; } if ( PM_InKnockDown( &gent->client->ps ) ) { return qfalse; } if ( (gent->s.numberclient->ps.groundEntityNum == ENTITYNUM_NONE //in air && ( (gent->client->ps.pm_flags&PMF_JUMPING) && gent->client->ps.velocity[2] > 0 )//jumped & going up or at water surface///*(gent->client->ps.waterHeightLevel==WHL_SHOULDERS&&gent->client->usercmd.upmove>0) ||*/ && gent->client->ps.forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_0 //force-jump capable && !(gent->client->ps.pm_flags&PMF_TRIGGER_PUSHED) )//not pushed by a trigger { if( gent->flags & FL_LOCK_PLAYER_WEAPONS ) // yes this locked weapons check also includes force powers, if we need a separate check later I'll make one { CG_PlayerLockedWeaponSpeech( qtrue ); return qfalse; } return qtrue; } return qfalse; } static void PM_JumpForDir( void ) { int anim = BOTH_JUMP1; if ( pm->cmd.forwardmove > 0 ) { anim = BOTH_JUMP1; pm->ps->pm_flags &= ~PMF_BACKWARDS_JUMP; } else if ( pm->cmd.forwardmove < 0 ) { anim = BOTH_JUMPBACK1; pm->ps->pm_flags |= PMF_BACKWARDS_JUMP; } else if ( pm->cmd.rightmove > 0 ) { anim = BOTH_JUMPRIGHT1; pm->ps->pm_flags &= ~PMF_BACKWARDS_JUMP; } else if ( pm->cmd.rightmove < 0 ) { anim = BOTH_JUMPLEFT1; pm->ps->pm_flags &= ~PMF_BACKWARDS_JUMP; } else { anim = BOTH_JUMP1; pm->ps->pm_flags &= ~PMF_BACKWARDS_JUMP; } if(!PM_InDeathAnim()) { PM_SetAnim(pm,SETANIM_LEGS,anim,SETANIM_FLAG_OVERRIDE, 100); // Only blend over 100ms } } qboolean PM_GentCantJump( gentity_t *gent ) {//FIXME: ugh, hacky, set a flag on NPC or something, please... if ( gent && gent->client && ( gent->client->NPC_class == CLASS_ATST || gent->client->NPC_class == CLASS_GONK || gent->client->NPC_class == CLASS_MARK1 || gent->client->NPC_class == CLASS_MARK2 || gent->client->NPC_class == CLASS_MOUSE || gent->client->NPC_class == CLASS_PROBE || gent->client->NPC_class == CLASS_PROTOCOL || gent->client->NPC_class == CLASS_R2D2 || gent->client->NPC_class == CLASS_R5D2 || gent->client->NPC_class == CLASS_SEEKER || gent->client->NPC_class == CLASS_REMOTE || gent->client->NPC_class == CLASS_SENTRY ) ) { return qtrue; } return qfalse; } static qboolean PM_CheckJump( void ) { //Don't allow jump until all buttons are up if ( pm->ps->pm_flags & PMF_RESPAWNED ) { return qfalse; } if ( PM_InKnockDown( pm->ps ) || PM_InRoll( pm->ps ) ) {//in knockdown return qfalse; } if ( PM_GentCantJump( pm->gent ) ) { return qfalse; } if ( PM_KickingAnim( pm->ps->legsAnim ) && !PM_InAirKickingAnim( pm->ps->legsAnim ) ) {//can't jump when in a kicking anim return qfalse; } /* if ( pm->cmd.buttons & BUTTON_FORCEJUMP ) { pm->ps->pm_flags |= PMF_JUMP_HELD; } */ #if METROID_JUMP if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && pm->gent && pm->gent->client && (pm->gent->client->NPC_class == CLASS_BOBAFETT || pm->gent->client->NPC_class == CLASS_ROCKETTROOPER) ) {//player playing as boba fett if ( pm->cmd.upmove > 0 ) {//turn on/go up if ( pm->ps->groundEntityNum == ENTITYNUM_NONE && !(pm->ps->pm_flags&PMF_JUMP_HELD) ) {//double-tap - must activate while in air if ( !JET_Flying( pm->gent ) ) { JET_FlyStart( pm->gent ); } } } else if ( pm->cmd.upmove < 0 ) {//turn it off (or should we just go down)? /* if ( JET_Flying( pm->gent ) ) { JET_FlyStop( pm->gent ); } */ } } else if ( pm->waterlevel < 3 )//|| (pm->ps->waterHeightLevel==WHL_SHOULDERS&&pm->cmd.upmove>0) ) { if ( pm->ps->gravity > 0 ) {//can't do this in zero-G if ( pm->ps->legsAnim == BOTH_FORCELONGLEAP_START || pm->ps->legsAnim == BOTH_FORCELONGLEAP_ATTACK || pm->ps->legsAnim == BOTH_FORCELONGLEAP_LAND ) {//in the middle of a force long-jump /* if ( pm->ps->groundEntityNum == ENTITYNUM_NONE && pm->cmd.upmove > 0 && pm->cmd.forwardmove > 0 ) */ if ( (pm->ps->legsAnim == BOTH_FORCELONGLEAP_START || pm->ps->legsAnim == BOTH_FORCELONGLEAP_ATTACK) && pm->ps->legsAnimTimer > 0 ) {//in the air //FIXME: need an actual set time so it doesn't matter when the attack happens //FIXME: make sure we don't jump further than force jump 3 allows vec3_t jFwdAngs, jFwdVec; VectorSet( jFwdAngs, 0, pm->ps->viewangles[YAW], 0 ); AngleVectors( jFwdAngs, jFwdVec, NULL, NULL ); float oldZVel = pm->ps->velocity[2]; if ( pm->ps->legsAnimTimer > 150 && oldZVel < 0 ) { oldZVel = 0; } VectorScale( jFwdVec, FORCE_LONG_LEAP_SPEED, pm->ps->velocity ); pm->ps->velocity[2] = oldZVel; pm->ps->pm_flags |= PMF_JUMP_HELD; pm->ps->pm_flags |= PMF_JUMPING|PMF_SLOW_MO_FALL; pm->ps->forcePowersActive |= (1<ps->legsAnim == BOTH_FORCELONGLEAP_START || pm->ps->legsAnim == BOTH_FORCELONGLEAP_ATTACK ) {//still in start anim, but it's run out pm->ps->forcePowersActive |= (1<ps->groundEntityNum == ENTITYNUM_NONE ) {//still in air? //hold it for another 50ms //PM_SetAnim( pm, SETANIM_BOTH, BOTH_FORCELONGLEAP_START, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); } } else {//in land-slide anim //FIXME: force some forward movement? Less if holding back? } if ( pm->ps->groundEntityNum == ENTITYNUM_NONE//still in air && pm->ps->origin[2] < pm->ps->jumpZStart )//dropped below original jump start {//slow down pm->ps->velocity[0] *= 0.75f; pm->ps->velocity[1] *= 0.75f; if ( (pm->ps->velocity[0]+pm->ps->velocity[1])*0.5f<=10.0f ) {//falling straight down PM_SetAnim( pm, SETANIM_BOTH, BOTH_FORCEINAIR1, SETANIM_FLAG_OVERRIDE ); } } return qfalse; } } else if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) //player-only for now && pm->cmd.upmove > 0 //trying to jump && pm->ps->forcePowerLevel[FP_LEVITATION] >= FORCE_LEVEL_3 //force jump 3 or better && pm->ps->forcePower >= FORCE_LONGJUMP_POWER //this costs 20 force to do && (pm->ps->forcePowersActive&(1<cmd.forwardmove > 0 //pushing forward && !pm->cmd.rightmove //not strafing && pm->ps->groundEntityNum != ENTITYNUM_NONE//not in mid-air && !(pm->ps->pm_flags&PMF_JUMP_HELD) //&& (float)(level.time-pm->ps->lastStationary) >= (3000.0f*g_timescale->value)//have to have a 3 second running start - relative to force speed slowdown && (level.time-pm->ps->forcePowerDebounce[FP_SPEED]) <= 250//have to have just started the force speed within the last half second && pm->gent ) {//start a force long-jump! vec3_t jFwdAngs, jFwdVec; //BOTH_FORCELONGLEAP_ATTACK if holding attack, too? PM_SetAnim( pm, SETANIM_BOTH, BOTH_FORCELONGLEAP_START, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); VectorSet( jFwdAngs, 0, pm->ps->viewangles[YAW], 0 ); AngleVectors( jFwdAngs, jFwdVec, NULL, NULL ); VectorScale( jFwdVec, FORCE_LONG_LEAP_SPEED, pm->ps->velocity ); pm->ps->velocity[2] = 320; pml.groundPlane = qfalse; pml.walking = qfalse; pm->ps->groundEntityNum = ENTITYNUM_NONE; pm->ps->jumpZStart = pm->ps->origin[2]; pm->ps->pm_flags |= PMF_JUMP_HELD; pm->ps->pm_flags |= PMF_JUMPING|PMF_SLOW_MO_FALL; //start force jump pm->ps->forcePowersActive |= (1<cmd.upmove = 0; // keep track of force jump stat if(pm->ps->clientNum == 0) { if( pm->gent && pm->gent->client ) { pm->gent->client->sess.missionStats.forceUsed[(int)FP_LEVITATION]++; } } G_SoundOnEnt( pm->gent, CHAN_BODY, "sound/weapons/force/jump.wav" ); WP_ForcePowerStop( pm->gent, FP_SPEED ); WP_ForcePowerDrain( pm->gent, FP_LEVITATION, FORCE_LONGJUMP_POWER );//drain the required force power G_StartMatrixEffect( pm->gent, 0, pm->ps->legsAnimTimer+500 ); return qtrue; } else if ( PM_InCartwheel( pm->ps->legsAnim ) || PM_InButterfly( pm->ps->legsAnim ) ) {//can't keep jumping up in cartwheels, ariels and butterflies } //FIXME: still able to pogo-jump... else if ( PM_ForceJumpingUp( pm->gent ) && (pm->ps->pm_flags&PMF_JUMP_HELD) )//||pm->ps->waterHeightLevel==WHL_SHOULDERS) ) {//force jumping && holding jump /* if ( !pm->ps->forceJumpZStart && (pm->ps->waterHeightLevel==WHL_SHOULDERS&&pm->cmd.upmove>0) ) { pm->ps->forceJumpZStart = pm->ps->origin[2]; } */ float curHeight = pm->ps->origin[2] - pm->ps->forceJumpZStart; //check for max force jump level and cap off & cut z vel if ( ( curHeight<=forceJumpHeight[0] ||//still below minimum jump height (pm->ps->forcePower&&pm->cmd.upmove>=10) ) &&////still have force power available and still trying to jump up curHeight < forceJumpHeight[pm->ps->forcePowerLevel[FP_LEVITATION]] )//still below maximum jump height {//can still go up //FIXME: after a certain amount of time of held jump, play force jump sound and flip if a dir is being held //FIXME: if hit a wall... should we cut velocity or allow them to slide up it? //FIXME: constantly drain force power at a rate by which the usage for maximum height would use up the full cost of force jump if ( curHeight > forceJumpHeight[0] ) {//passed normal jump height *2? if ( !(pm->ps->forcePowersActive&(1<ps->forcePowersActive |= (1<gent ) { G_SoundOnEnt( pm->gent, CHAN_BODY, "sound/weapons/force/jump.wav" ); // keep track of force jump stat if(pm->ps->clientNum == 0 && pm->gent->client) { pm->gent->client->sess.missionStats.forceUsed[(int)FP_LEVITATION]++; } } //play flip //FIXME: do this only when they stop the jump (below) or when they're just about to hit the peak of the jump if ( PM_InAirKickingAnim( pm->ps->legsAnim ) && pm->ps->legsAnimTimer ) {//still in kick } else if ((pm->cmd.forwardmove || pm->cmd.rightmove) && //pushing in a dir //pm->ps->legsAnim != BOTH_ARIAL_F1 &&//not already flipping pm->ps->legsAnim != BOTH_FLIP_F && pm->ps->legsAnim != BOTH_FLIP_B && pm->ps->legsAnim != BOTH_FLIP_R && pm->ps->legsAnim != BOTH_FLIP_L && pm->ps->legsAnim != BOTH_ALORA_FLIP_1 && pm->ps->legsAnim != BOTH_ALORA_FLIP_2 && pm->ps->legsAnim != BOTH_ALORA_FLIP_3 && cg.renderingThirdPerson//third person only && !cg.zoomMode //not zoomed in && !(pm->ps->saber[0].saberFlags&SFL_NO_FLIPS)//okay to do flips with this saber && (!pm->ps->dualSabers || !(pm->ps->saber[1].saberFlags&SFL_NO_FLIPS) )//okay to do flips with this saber ) {//FIXME: this could end up playing twice if the jump is very long... int anim = BOTH_FORCEINAIR1; int parts = SETANIM_BOTH; if ( pm->cmd.forwardmove > 0 ) { /* if ( pm->ps->forcePowerLevel[FP_LEVITATION] < FORCE_LEVEL_2 ) { anim = BOTH_ARIAL_F1; } else */ if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_ALORA && Q_irand( 0, 3 ) ) { anim = Q_irand( BOTH_ALORA_FLIP_1, BOTH_ALORA_FLIP_3 ); } else { anim = BOTH_FLIP_F; } } else if ( pm->cmd.forwardmove < 0 ) { anim = BOTH_FLIP_B; } else if ( pm->cmd.rightmove > 0 ) { anim = BOTH_FLIP_R; } else if ( pm->cmd.rightmove < 0 ) { anim = BOTH_FLIP_L; } if ( pm->ps->weaponTime ) {//FIXME: really only care if we're in a saber attack anim... parts = SETANIM_LEGS; } PM_SetAnim( pm, parts, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); } else if ( pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_1 ) {//FIXME: really want to know how far off ground we are, probably... vec3_t facingFwd, facingRight, facingAngles = {0, pm->ps->viewangles[YAW], 0}; int anim = -1; AngleVectors( facingAngles, facingFwd, facingRight, NULL ); float dotR = DotProduct( facingRight, pm->ps->velocity ); float dotF = DotProduct( facingFwd, pm->ps->velocity ); if ( fabs(dotR) > fabs(dotF) * 1.5 ) { if ( dotR > 150 ) { anim = BOTH_FORCEJUMPRIGHT1; } else if ( dotR < -150 ) { anim = BOTH_FORCEJUMPLEFT1; } } else { if ( dotF > 150 ) { anim = BOTH_FORCEJUMP1; } else if ( dotF < -150 ) { anim = BOTH_FORCEJUMPBACK1; } } if ( anim != -1 ) { int parts = SETANIM_BOTH; if ( pm->ps->weaponTime ) {//FIXME: really only care if we're in a saber attack anim... parts = SETANIM_LEGS; } PM_SetAnim( pm, parts, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); } } } else { if ( !pm->ps->legsAnimTimer ) {//not in the middle of a legsAnim int anim = pm->ps->legsAnim; int newAnim = -1; switch ( anim ) { case BOTH_FORCEJUMP1: newAnim = BOTH_FORCELAND1;//BOTH_FORCEINAIR1; break; case BOTH_FORCEJUMPBACK1: newAnim = BOTH_FORCELANDBACK1;//BOTH_FORCEINAIRBACK1; break; case BOTH_FORCEJUMPLEFT1: newAnim = BOTH_FORCELANDLEFT1;//BOTH_FORCEINAIRLEFT1; break; case BOTH_FORCEJUMPRIGHT1: newAnim = BOTH_FORCELANDRIGHT1;//BOTH_FORCEINAIRRIGHT1; break; } if ( newAnim != -1 ) { int parts = SETANIM_BOTH; if ( pm->ps->weaponTime ) {//FIXME: really only care if we're in a saber attack anim... parts = SETANIM_LEGS; } PM_SetAnim( pm, parts, newAnim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); } } } } //need to scale this down, start with height velocity (based on max force jump height) and scale down to regular jump vel pm->ps->velocity[2] = (forceJumpHeight[pm->ps->forcePowerLevel[FP_LEVITATION]]-curHeight)/forceJumpHeight[pm->ps->forcePowerLevel[FP_LEVITATION]]*forceJumpStrength[pm->ps->forcePowerLevel[FP_LEVITATION]];//JUMP_VELOCITY; pm->ps->velocity[2] /= 10; pm->ps->velocity[2] += JUMP_VELOCITY; pm->ps->pm_flags |= PMF_JUMP_HELD; } else if ( curHeight > forceJumpHeight[0] && curHeight < forceJumpHeight[pm->ps->forcePowerLevel[FP_LEVITATION]] - forceJumpHeight[0] ) {//still have some headroom, don't totally stop it if ( pm->ps->velocity[2] > JUMP_VELOCITY ) { pm->ps->velocity[2] = JUMP_VELOCITY; } } else { pm->ps->velocity[2] = 0; } pm->cmd.upmove = 0; return qfalse; } } } #endif //Not jumping if ( pm->cmd.upmove < 10 ) { return qfalse; } // must wait for jump to be released if ( pm->ps->pm_flags & PMF_JUMP_HELD ) { // clear upmove so cmdscale doesn't lower running speed pm->cmd.upmove = 0; return qfalse; } if ( pm->ps->gravity <= 0 ) {//in low grav, you push in the dir you're facing as long as there is something behind you to shove off of vec3_t forward, back; trace_t trace; AngleVectors( pm->ps->viewangles, forward, NULL, NULL ); VectorMA( pm->ps->origin, -8, forward, back ); pm->trace( &trace, pm->ps->origin, pm->mins, pm->maxs, back, pm->ps->clientNum, pm->tracemask&~(CONTENTS_PLAYERCLIP|CONTENTS_MONSTERCLIP), (EG2_Collision)0, 0 ); pm->cmd.upmove = 0; if ( trace.fraction < 1.0f ) { VectorMA( pm->ps->velocity, JUMP_VELOCITY/2, forward, pm->ps->velocity ); //FIXME: kicking off wall anim? At least check what anim we're in? PM_SetAnim(pm,SETANIM_LEGS,BOTH_FORCEJUMP1,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD|SETANIM_FLAG_RESTART); } else {//else no surf close enough to push off of return qfalse; } if ( pm->ps->groundEntityNum == ENTITYNUM_NONE ) {//need to set some things and return //Jumping pm->ps->forceJumpZStart = 0; pml.groundPlane = qfalse; pml.walking = qfalse; pm->ps->pm_flags |= (PMF_JUMPING|PMF_JUMP_HELD); pm->ps->groundEntityNum = ENTITYNUM_NONE; pm->ps->jumpZStart = pm->ps->origin[2]; if ( pm->gent ) { if ( !Q3_TaskIDPending( pm->gent, TID_CHAN_VOICE ) ) { PM_AddEvent( EV_JUMP ); } } else { PM_AddEvent( EV_JUMP ); } return qtrue; }//else no surf close enough to push off of } else if ( pm->cmd.upmove > 0 //want to jump && pm->waterlevel < 2 //not in water above ankles && pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_0 //have force jump ability && !(pm->ps->pm_flags&PMF_JUMP_HELD)//not holding jump from a previous jump //&& !PM_InKnockDown( pm->ps )//not in a knockdown && pm->ps->forceRageRecoveryTime < pm->cmd.serverTime //not in a force Rage recovery period && pm->gent && WP_ForcePowerAvailable( pm->gent, FP_LEVITATION, 0 ) //have enough force power to jump && ((pm->ps->clientNum&&!PM_ControlledByPlayer())||((pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && cg.renderingThirdPerson && !cg.zoomMode && !(pm->gent->flags&FL_LOCK_PLAYER_WEAPONS) )) )// yes this locked weapons check also includes force powers, if we need a separate check later I'll make one { if ( pm->gent->NPC && pm->gent->NPC->rank != RANK_CREWMAN && pm->gent->NPC->rank <= RANK_LT_JG ) {//reborn who are not acrobats can't do any of these acrobatics //FIXME: extern these abilities in the .npc file! } else if ( pm->ps->groundEntityNum != ENTITYNUM_NONE ) {//on the ground //check for left-wall and right-wall special jumps int anim = -1; float vertPush = 0; int forcePowerCostOverride = 0; // Cartwheels/ariels/butterflies if ( (pm->ps->weapon==WP_SABER&&G_TryingCartwheel(pm->gent,&pm->cmd)/*(pm->cmd.buttons&BUTTON_FORCE_FOCUS)*/&&(pm->cmd.buttons&BUTTON_ATTACK))//using saber and holding focus + attack // ||(pm->ps->weapon!=WP_SABER&&((pm->cmd.buttons&BUTTON_ATTACK)||(pm->cmd.buttons&BUTTON_ALT_ATTACK)) ) )//using any other weapon and hitting either attack button && (((pm->ps->clientNum>=MAX_CLIENTS&&!PM_ControlledByPlayer())&&pm->cmd.upmove > 0&& pm->ps->velocity[2] >= 0 )//jumping NPC, going up already ||((pm->ps->clientNumgent,&pm->cmd)/*(pm->cmd.buttons&BUTTON_FORCE_FOCUS)*/))//focus-holding player && G_EnoughPowerForSpecialMove( pm->ps->forcePower, SABER_ALT_ATTACK_POWER_LR )/*pm->ps->forcePower >= SABER_ALT_ATTACK_POWER_LR*/ )// have enough power {//holding attack and jumping if ( pm->cmd.rightmove > 0 ) { // If they're using the staff we do different anims. if ( pm->ps->saberAnimLevel == SS_STAFF && pm->ps->weapon == WP_SABER ) { if ( (pm->ps->clientNum >= MAX_CLIENTS&&!PM_ControlledByPlayer()) || pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_2 ) { anim = BOTH_BUTTERFLY_RIGHT; forcePowerCostOverride = G_CostForSpecialMove( SABER_ALT_ATTACK_POWER_LR ); } } else if ( (pm->ps->clientNum >= MAX_CLIENTS&&!PM_ControlledByPlayer()) || pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_1 ) { if ( !(pm->ps->saber[0].saberFlags&SFL_NO_CARTWHEELS) && (!pm->ps->dualSabers || !(pm->ps->saber[1].saberFlags&SFL_NO_CARTWHEELS)) ) {//okay to do cartwheels with this saber if ( pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer() ) {//player: since we're on the ground, always do a cartwheel /* anim = BOTH_CARTWHEEL_RIGHT; forcePowerCostOverride = G_CostForSpecialMove( SABER_ALT_ATTACK_POWER_LR ); */ } else { vertPush = JUMP_VELOCITY; if ( Q_irand( 0, 1 ) ) { anim = BOTH_ARIAL_RIGHT; forcePowerCostOverride = G_CostForSpecialMove( SABER_ALT_ATTACK_POWER_LR ); } else { anim = BOTH_CARTWHEEL_RIGHT; forcePowerCostOverride = G_CostForSpecialMove( SABER_ALT_ATTACK_POWER_LR ); } } } } } else if ( pm->cmd.rightmove < 0 ) { // If they're using the staff we do different anims. if ( pm->ps->saberAnimLevel == SS_STAFF && pm->ps->weapon == WP_SABER ) { if ( (pm->ps->clientNum >= MAX_CLIENTS&&!PM_ControlledByPlayer()) || pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_2 ) { anim = BOTH_BUTTERFLY_LEFT; forcePowerCostOverride = G_CostForSpecialMove( SABER_ALT_ATTACK_POWER_LR ); } } else if ( (pm->ps->clientNum >= MAX_CLIENTS&&!PM_ControlledByPlayer()) || pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_1 ) { if ( !(pm->ps->saber[0].saberFlags&SFL_NO_CARTWHEELS) && (!pm->ps->dualSabers || !(pm->ps->saber[1].saberFlags&SFL_NO_CARTWHEELS)) ) {//okay to do cartwheels with this saber if ( pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer() ) {//player: since we're on the ground, always do a cartwheel /* anim = BOTH_CARTWHEEL_LEFT; forcePowerCostOverride = G_CostForSpecialMove( SABER_ALT_ATTACK_POWER_LR ); */ } else { vertPush = JUMP_VELOCITY; if ( Q_irand( 0, 1 ) ) { anim = BOTH_ARIAL_LEFT; forcePowerCostOverride = G_CostForSpecialMove( SABER_ALT_ATTACK_POWER_LR ); } else { anim = BOTH_CARTWHEEL_LEFT; forcePowerCostOverride = G_CostForSpecialMove( SABER_ALT_ATTACK_POWER_LR ); } } } } } } else if ( pm->cmd.rightmove > 0 && pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_1 ) {//strafing right if ( pm->cmd.forwardmove > 0 ) {//wall-run if ( !(pm->ps->saber[0].saberFlags&SFL_NO_WALL_RUNS) && (!pm->ps->dualSabers || !(pm->ps->saber[1].saberFlags&SFL_NO_WALL_RUNS)) ) {//okay to do wall-runs with this saber vertPush = forceJumpStrength[FORCE_LEVEL_2]/2.0f; anim = BOTH_WALL_RUN_RIGHT; } } else if ( pm->cmd.forwardmove == 0 ) {//wall-flip if ( !(pm->ps->saber[0].saberFlags&SFL_NO_WALL_FLIPS) && (!pm->ps->dualSabers || !(pm->ps->saber[1].saberFlags&SFL_NO_WALL_FLIPS)) ) {//okay to do wall-flips with this saber vertPush = forceJumpStrength[FORCE_LEVEL_2]/2.25f; anim = BOTH_WALL_FLIP_RIGHT; } } } else if ( pm->cmd.rightmove < 0 && pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_1 ) {//strafing left if ( pm->cmd.forwardmove > 0 ) {//wall-run if ( !(pm->ps->saber[0].saberFlags&SFL_NO_WALL_RUNS) && (!pm->ps->dualSabers || !(pm->ps->saber[1].saberFlags&SFL_NO_WALL_RUNS)) ) {//okay to do wall-runs with this saber vertPush = forceJumpStrength[FORCE_LEVEL_2]/2.0f; anim = BOTH_WALL_RUN_LEFT; } } else if ( pm->cmd.forwardmove == 0 ) {//wall-flip if ( !(pm->ps->saber[0].saberFlags&SFL_NO_WALL_FLIPS) && (!pm->ps->dualSabers || !(pm->ps->saber[1].saberFlags&SFL_NO_WALL_FLIPS)) ) {//okay to do wall-flips with this saber vertPush = forceJumpStrength[FORCE_LEVEL_2]/2.25f; anim = BOTH_WALL_FLIP_LEFT; } } } else if ( /*pm->ps->clientNum >= MAX_CLIENTS//not the player && !PM_ControlledByPlayer() //not controlled by player &&*/ pm->cmd.forwardmove > 0 //pushing forward && pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_1 )//have jump 2 or higher {//step off wall, flip backwards if ( VectorLengthSquared( pm->ps->velocity ) > 40000 /*200*200*/) {//have to be moving... FIXME: make sure it's opposite the wall... or at least forward? if ( pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_2 ) {//run all the way up wwall if ( !(pm->ps->saber[0].saberFlags&SFL_NO_WALL_RUNS) && (!pm->ps->dualSabers || !(pm->ps->saber[1].saberFlags&SFL_NO_WALL_RUNS)) ) {//okay to do wall-runs with this saber vertPush = forceJumpStrength[FORCE_LEVEL_2]/2.0f; anim = BOTH_FORCEWALLRUNFLIP_START; } } else {//run just a couple steps up if ( !(pm->ps->saber[0].saberFlags&SFL_NO_WALL_FLIPS) && (!pm->ps->dualSabers || !(pm->ps->saber[1].saberFlags&SFL_NO_WALL_FLIPS)) ) {//okay to do wall-flips with this saber vertPush = forceJumpStrength[FORCE_LEVEL_2]/2.25f; anim = BOTH_WALL_FLIP_BACK1; } } } } else if ( pm->cmd.forwardmove < 0 //pushing back //&& pm->ps->clientNum//not the player && !(pm->cmd.buttons&BUTTON_ATTACK) )//not attacking {//back-jump does backflip... FIXME: always?! What about just doing a normal jump backwards? if ( pm->ps->velocity[2] >= 0 ) {//must be going up already if ( !(pm->ps->saber[0].saberFlags&SFL_NO_FLIPS) && (!pm->ps->dualSabers || !(pm->ps->saber[1].saberFlags&SFL_NO_FLIPS)) ) {//okay to do backstabs with this saber vertPush = JUMP_VELOCITY; if ( pm->gent->client && pm->gent->client->NPC_class == CLASS_ALORA && !Q_irand( 0, 2 ) ) { anim = BOTH_ALORA_FLIP_B; } else { anim = PM_PickAnim( pm->gent, BOTH_FLIP_BACK1, BOTH_FLIP_BACK3 ); } } } } else if ( VectorLengthSquared( pm->ps->velocity ) < 256 /*16 squared*/) {//not moving if ( pm->ps->weapon == WP_SABER && (pm->cmd.buttons & BUTTON_ATTACK) ) { saberMoveName_t overrideJumpAttackUpMove = LS_INVALID; if ( pm->ps->saber[0].jumpAtkUpMove != LS_INVALID ) { if ( pm->ps->saber[0].jumpAtkUpMove != LS_NONE ) {//actually overriding overrideJumpAttackUpMove = (saberMoveName_t)pm->ps->saber[0].jumpAtkUpMove; } else if ( pm->ps->dualSabers && pm->ps->saber[1].jumpAtkUpMove > LS_NONE ) {//would be cancelling it, but check the second saber, too overrideJumpAttackUpMove = (saberMoveName_t)pm->ps->saber[1].jumpAtkUpMove; } else {//nope, just cancel it overrideJumpAttackUpMove = LS_NONE; } } else if ( pm->ps->dualSabers && pm->ps->saber[1].jumpAtkUpMove != LS_INVALID ) {//first saber not overridden, check second overrideJumpAttackUpMove = (saberMoveName_t)pm->ps->saber[0].jumpAtkUpMove; } if ( overrideJumpAttackUpMove != LS_INVALID ) {//do this move instead if ( overrideJumpAttackUpMove != LS_NONE ) { anim = saberMoveData[overrideJumpAttackUpMove].animToUse; } } else if ( pm->ps->saberAnimLevel == SS_MEDIUM ) { /* //Only tavion does these now if ( pm->ps->clientNum && Q_irand( 0, 1 ) ) {//butterfly... FIXME: does direction matter? vertPush = JUMP_VELOCITY; if ( Q_irand( 0, 1 ) ) { anim = BOTH_BUTTERFLY_LEFT; } else { anim = BOTH_BUTTERFLY_RIGHT; } } else */if ( pm->ps->clientNum >= MAX_CLIENTS && !PM_ControlledByPlayer() )//NOTE: pretty much useless, so player never does these {//jump-spin FIXME: does direction matter? vertPush = forceJumpStrength[FORCE_LEVEL_2]/1.5f; if ( pm->gent->client && pm->gent->client->NPC_class == CLASS_ALORA ) { anim = BOTH_ALORA_SPIN; } else { anim = Q_irand( BOTH_FJSS_TR_BL, BOTH_FJSS_TL_BR ); } } } } } if ( anim != -1 && PM_HasAnimation( pm->gent, anim ) ) { vec3_t fwd, right, traceto, mins = {pm->mins[0],pm->mins[1],0}, maxs = {pm->maxs[0],pm->maxs[1],24}, fwdAngles = {0, pm->ps->viewangles[YAW], 0}; trace_t trace; qboolean doTrace = qfalse; int contents = CONTENTS_SOLID; AngleVectors( fwdAngles, fwd, right, NULL ); //trace-check for a wall, if necc. switch ( anim ) { case BOTH_WALL_FLIP_LEFT: if ( g_debugMelee->integer ) { contents |= CONTENTS_BODY; } //NOTE: purposely falls through to next case! case BOTH_WALL_RUN_LEFT: doTrace = qtrue; VectorMA( pm->ps->origin, -16, right, traceto ); break; case BOTH_WALL_FLIP_RIGHT: if ( g_debugMelee->integer ) { contents |= CONTENTS_BODY; } //NOTE: purposely falls through to next case! case BOTH_WALL_RUN_RIGHT: doTrace = qtrue; VectorMA( pm->ps->origin, 16, right, traceto ); break; case BOTH_WALL_FLIP_BACK1: if ( g_debugMelee->integer ) { contents |= CONTENTS_BODY; } doTrace = qtrue; VectorMA( pm->ps->origin, 32, fwd, traceto );//was 16 break; case BOTH_FORCEWALLRUNFLIP_START: if ( g_debugMelee->integer ) { contents |= CONTENTS_BODY; } doTrace = qtrue; VectorMA( pm->ps->origin, 32, fwd, traceto );//was 16 break; } vec3_t idealNormal={0}, wallNormal={0}; if ( doTrace ) { //FIXME: all these jump ones should check for head clearance pm->trace( &trace, pm->ps->origin, mins, maxs, traceto, pm->ps->clientNum, contents, (EG2_Collision)0, 0 ); VectorCopy( trace.plane.normal, wallNormal ); VectorNormalize( wallNormal ); VectorSubtract( pm->ps->origin, traceto, idealNormal ); VectorNormalize( idealNormal ); if ( anim == BOTH_WALL_FLIP_LEFT ) {//sigh.. check for bottomless pit to the right trace_t trace2; vec3_t start; VectorMA( pm->ps->origin, 128, right, traceto ); pm->trace( &trace2, pm->ps->origin, mins, maxs, traceto, pm->ps->clientNum, contents, (EG2_Collision)0, 0 ); if ( !trace2.allsolid && !trace2.startsolid ) { VectorCopy( trace2.endpos, traceto ); VectorCopy( traceto, start ); traceto[2] -= 384; pm->trace( &trace2, start, mins, maxs, traceto, pm->ps->clientNum, contents, (EG2_Collision)0, 0 ); if ( !trace2.allsolid && !trace2.startsolid && trace2.fraction >= 1.0f ) {//bottomless pit! trace.fraction = 1.0f;//way to stop it from doing the side-flip } } } else if ( anim == BOTH_WALL_FLIP_RIGHT ) {//sigh.. check for bottomless pit to the left trace_t trace2; vec3_t start; VectorMA( pm->ps->origin, -128, right, traceto ); pm->trace( &trace2, pm->ps->origin, mins, maxs, traceto, pm->ps->clientNum, contents, (EG2_Collision)0, 0 ); if ( !trace2.allsolid && !trace2.startsolid ) { VectorCopy( trace2.endpos, traceto ); VectorCopy( traceto, start ); traceto[2] -= 384; pm->trace( &trace2, start, mins, maxs, traceto, pm->ps->clientNum, contents, (EG2_Collision)0, 0 ); if ( !trace2.allsolid && !trace2.startsolid && trace2.fraction >= 1.0f ) {//bottomless pit! trace.fraction = 1.0f;//way to stop it from doing the side-flip } } } else { if ( anim == BOTH_WALL_FLIP_BACK1 || anim == BOTH_FORCEWALLRUNFLIP_START ) {//trace up and forward a little to make sure the wall it at least 64 tall if ( (contents&CONTENTS_BODY)//included entitied && (trace.contents&CONTENTS_BODY) //hit an entity && g_entities[trace.entityNum].client )//hit a client {//no need to trace up, it's all good... if ( PM_InOnGroundAnim( &g_entities[trace.entityNum].client->ps ) )//on the ground, no jump {//can't jump off guys on ground trace.fraction = 1.0f;//way to stop if from doing the jump } else if ( anim == BOTH_FORCEWALLRUNFLIP_START ) {//instead of wall-running up, do the backflip anim = BOTH_WALL_FLIP_BACK1; vertPush = forceJumpStrength[FORCE_LEVEL_2]/2.25f; } } else if ( anim == BOTH_WALL_FLIP_BACK1 ) { trace_t trace2; vec3_t start; VectorCopy( pm->ps->origin, start ); start[2] += 64; VectorMA( start, 32, fwd, traceto ); pm->trace( &trace2, start, mins, maxs, traceto, pm->ps->clientNum, contents, (EG2_Collision)0, 0 ); if ( trace2.allsolid || trace2.startsolid || trace2.fraction >= 1.0f ) {//no room above or no wall in front at that height trace.fraction = 1.0f;//way to stop if from doing the jump } } } if ( trace.fraction < 1.0f ) {//still valid to jump if ( anim == BOTH_WALL_FLIP_BACK1 ) {//sigh.. check for bottomless pit to the rear trace_t trace2; vec3_t start; VectorMA( pm->ps->origin, -128, fwd, traceto ); pm->trace( &trace2, pm->ps->origin, mins, maxs, traceto, pm->ps->clientNum, contents, (EG2_Collision)0, 0 ); if ( !trace2.allsolid && !trace2.startsolid ) { VectorCopy( trace2.endpos, traceto ); VectorCopy( traceto, start ); traceto[2] -= 384; pm->trace( &trace2, start, mins, maxs, traceto, pm->ps->clientNum, contents, (EG2_Collision)0, 0 ); if ( !trace2.allsolid && !trace2.startsolid && trace2.fraction >= 1.0f ) {//bottomless pit! trace.fraction = 1.0f;//way to stop it from doing the side-flip } } } } } } gentity_t *traceEnt = &g_entities[trace.entityNum]; if ( !doTrace || (trace.fraction < 1.0f&&((trace.entityNums.solid!=SOLID_BMODEL)||DotProduct(wallNormal,idealNormal)>0.7)) ) {//there is a wall there if ( (anim != BOTH_WALL_RUN_LEFT && anim != BOTH_WALL_RUN_RIGHT && anim != BOTH_FORCEWALLRUNFLIP_START) || (wallNormal[2] >= 0.0f && wallNormal[2] <= MAX_WALL_RUN_Z_NORMAL) ) {//wall-runs can only run on relatively flat walls, sorry. if ( anim == BOTH_ARIAL_LEFT || anim == BOTH_CARTWHEEL_LEFT ) { pm->ps->velocity[0] = pm->ps->velocity[1] = 0; VectorMA( pm->ps->velocity, -185, right, pm->ps->velocity ); } else if ( anim == BOTH_ARIAL_RIGHT || anim == BOTH_CARTWHEEL_RIGHT ) { pm->ps->velocity[0] = pm->ps->velocity[1] = 0; VectorMA( pm->ps->velocity, 185, right, pm->ps->velocity ); } else if ( anim == BOTH_BUTTERFLY_LEFT ) { pm->ps->velocity[0] = pm->ps->velocity[1] = 0; VectorMA( pm->ps->velocity, -190, right, pm->ps->velocity ); } else if ( anim == BOTH_BUTTERFLY_RIGHT ) { pm->ps->velocity[0] = pm->ps->velocity[1] = 0; VectorMA( pm->ps->velocity, 190, right, pm->ps->velocity ); } //move me to side else if ( anim == BOTH_WALL_FLIP_LEFT ) { pm->ps->velocity[0] = pm->ps->velocity[1] = 0; VectorMA( pm->ps->velocity, 150, right, pm->ps->velocity ); } else if ( anim == BOTH_WALL_FLIP_RIGHT ) { pm->ps->velocity[0] = pm->ps->velocity[1] = 0; VectorMA( pm->ps->velocity, -150, right, pm->ps->velocity ); } else if ( anim == BOTH_FLIP_BACK1 || anim == BOTH_FLIP_BACK2 || anim == BOTH_FLIP_BACK3 || anim == BOTH_ALORA_FLIP_B || anim == BOTH_WALL_FLIP_BACK1 ) { pm->ps->velocity[0] = pm->ps->velocity[1] = 0; VectorMA( pm->ps->velocity, -150, fwd, pm->ps->velocity ); } //kick if jumping off an ent if ( doTrace && anim != BOTH_WALL_RUN_LEFT && anim != BOTH_WALL_RUN_RIGHT && anim != BOTH_FORCEWALLRUNFLIP_START) { if ( pm->gent && trace.entityNum < ENTITYNUM_WORLD ) { if ( traceEnt && traceEnt->client && traceEnt->health > 0 && traceEnt->takedamage && traceEnt->client->NPC_class != CLASS_GALAKMECH && traceEnt->client->NPC_class != CLASS_DESANN && !(traceEnt->flags&FL_NO_KNOCKBACK) ) {//push them away and do pain vec3_t oppDir, fxDir; float strength = VectorNormalize2( pm->ps->velocity, oppDir ); VectorScale( oppDir, -1, oppDir ); //FIXME: need knockdown anim G_Damage( traceEnt, pm->gent, pm->gent, oppDir, traceEnt->currentOrigin, 10, DAMAGE_NO_ARMOR|DAMAGE_NO_HIT_LOC|DAMAGE_NO_KNOCKBACK, MOD_MELEE ); VectorCopy( fwd, fxDir ); VectorScale( fxDir, -1, fxDir ); G_PlayEffect( G_EffectIndex( "melee/kick_impact" ), trace.endpos, fxDir ); //G_Sound( traceEnt, G_SoundIndex( va("sound/weapons/melee/punch%d", Q_irand(1, 4)) ) ); if ( traceEnt->health > 0 ) {//didn't kill him if ( (traceEnt->s.number==0&&!Q_irand(0,g_spskill->integer)) || (traceEnt->NPC!=NULL&&Q_irand(RANK_CIVILIAN,traceEnt->NPC->rank)+Q_irand(-2,2)ps->velocity[2] = vertPush; } //animate me if ( anim == BOTH_BUTTERFLY_RIGHT ) { PM_SetSaberMove( LS_BUTTERFLY_RIGHT ); } else if ( anim == BOTH_BUTTERFLY_LEFT ) { PM_SetSaberMove( LS_BUTTERFLY_LEFT ); } else {//not a proper saberMove, so do set all the details manually int parts = SETANIM_LEGS; if ( /*anim == BOTH_BUTTERFLY_LEFT || anim == BOTH_BUTTERFLY_RIGHT ||*/ anim == BOTH_FJSS_TR_BL || anim == BOTH_FJSS_TL_BR ) { parts = SETANIM_BOTH; pm->cmd.buttons&=~BUTTON_ATTACK; pm->ps->saberMove = LS_NONE; pm->gent->client->ps.SaberActivateTrail( 300 ); } else if ( !pm->ps->weaponTime ) { parts = SETANIM_BOTH; } PM_SetAnim( pm, parts, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD, 0 ); if ( /*anim == BOTH_BUTTERFLY_LEFT || anim == BOTH_BUTTERFLY_RIGHT ||*/ anim == BOTH_FJSS_TR_BL || anim == BOTH_FJSS_TL_BR || anim == BOTH_FORCEWALLRUNFLIP_START ) { pm->ps->weaponTime = pm->ps->torsoAnimTimer; } else if ( anim == BOTH_WALL_FLIP_LEFT || anim == BOTH_WALL_FLIP_RIGHT || anim == BOTH_WALL_FLIP_BACK1 ) {//let us do some more moves after this pm->ps->saberAttackChainCount = 0; } } pm->ps->forceJumpZStart = pm->ps->origin[2];//so we don't take damage if we land at same height pm->ps->pm_flags |= (PMF_JUMPING|PMF_SLOW_MO_FALL); pm->cmd.upmove = 0; G_SoundOnEnt( pm->gent, CHAN_BODY, "sound/weapons/force/jump.wav" ); WP_ForcePowerDrain( pm->gent, FP_LEVITATION, forcePowerCostOverride ); } } } } else {//in the air int legsAnim = pm->ps->legsAnim; if ( legsAnim == BOTH_WALL_RUN_LEFT || legsAnim == BOTH_WALL_RUN_RIGHT ) {//running on a wall vec3_t right, traceto, mins = {pm->mins[0],pm->mins[0],0}, maxs = {pm->maxs[0],pm->maxs[0],24}, fwdAngles = {0, pm->ps->viewangles[YAW], 0}; trace_t trace; int anim = -1; AngleVectors( fwdAngles, NULL, right, NULL ); if ( legsAnim == BOTH_WALL_RUN_LEFT ) { if ( pm->ps->legsAnimTimer > 400 ) {//not at the end of the anim float animLen = PM_AnimLength( pm->gent->client->clientInfo.animFileIndex, BOTH_WALL_RUN_LEFT ); if ( pm->ps->legsAnimTimer < animLen - 400 ) {//not at start of anim VectorMA( pm->ps->origin, -16, right, traceto ); anim = BOTH_WALL_RUN_LEFT_FLIP; } } } else if ( legsAnim == BOTH_WALL_RUN_RIGHT ) { if ( pm->ps->legsAnimTimer > 400 ) {//not at the end of the anim float animLen = PM_AnimLength( pm->gent->client->clientInfo.animFileIndex, BOTH_WALL_RUN_RIGHT ); if ( pm->ps->legsAnimTimer < animLen - 400 ) {//not at start of anim VectorMA( pm->ps->origin, 16, right, traceto ); anim = BOTH_WALL_RUN_RIGHT_FLIP; } } } if ( anim != -1 ) { pm->trace( &trace, pm->ps->origin, mins, maxs, traceto, pm->ps->clientNum, CONTENTS_SOLID|CONTENTS_BODY, (EG2_Collision)0, 0 ); if ( trace.fraction < 1.0f ) {//flip off wall if ( anim == BOTH_WALL_RUN_LEFT_FLIP ) { pm->ps->velocity[0] *= 0.5f; pm->ps->velocity[1] *= 0.5f; VectorMA( pm->ps->velocity, 150, right, pm->ps->velocity ); } else if ( anim == BOTH_WALL_RUN_RIGHT_FLIP ) { pm->ps->velocity[0] *= 0.5f; pm->ps->velocity[1] *= 0.5f; VectorMA( pm->ps->velocity, -150, right, pm->ps->velocity ); } PM_SetAnim( pm, SETANIM_BOTH, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD, 0 ); pm->ps->pm_flags |= PMF_JUMPING|PMF_SLOW_MO_FALL; pm->cmd.upmove = 0; } } if ( pm->cmd.upmove != 0 ) {//jump failed, so don't try to do normal jump code, just return return qfalse; } } else if ( pm->ps->legsAnim == BOTH_FORCEWALLRUNFLIP_START ) {//want to jump off wall vec3_t fwd, traceto, mins = {pm->mins[0],pm->mins[0],0}, maxs = {pm->maxs[0],pm->maxs[0],24}, fwdAngles = {0, pm->ps->viewangles[YAW], 0}; trace_t trace; int anim = -1; AngleVectors( fwdAngles, fwd, NULL, NULL ); float animLen = PM_AnimLength( pm->gent->client->clientInfo.animFileIndex, BOTH_FORCEWALLRUNFLIP_START ); if ( pm->ps->legsAnimTimer < animLen - 250 )//was 400 {//not at start of anim VectorMA( pm->ps->origin, 16, fwd, traceto ); anim = BOTH_FORCEWALLRUNFLIP_END; } if ( anim != -1 ) { pm->trace( &trace, pm->ps->origin, mins, maxs, traceto, pm->ps->clientNum, CONTENTS_SOLID|CONTENTS_BODY, (EG2_Collision)0, 0 ); if ( trace.fraction < 1.0f ) {//flip off wall pm->ps->velocity[0] *= 0.5f; pm->ps->velocity[1] *= 0.5f; VectorMA( pm->ps->velocity, WALL_RUN_UP_BACKFLIP_SPEED, fwd, pm->ps->velocity ); pm->ps->velocity[2] += 200; int parts = SETANIM_LEGS; if ( !pm->ps->weaponTime ) {//not attacking, set anim on both parts = SETANIM_BOTH; } PM_SetAnim( pm, parts, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD, 0 ); //FIXME: do damage to traceEnt, like above? pm->ps->pm_flags |= PMF_JUMPING|PMF_SLOW_MO_FALL; pm->cmd.upmove = 0; PM_AddEvent( EV_JUMP ); } } if ( pm->cmd.upmove != 0 ) {//jump failed, so don't try to do normal jump code, just return return qfalse; } } /* else if ( pm->cmd.forwardmove < 0 && pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_1 && !(pm->ps->pm_flags&PMF_JUMP_HELD) //not holding jump && (level.time - pm->ps->lastOnGround) <= 250 //just jumped )//&& !(pm->cmd.buttons&BUTTON_ATTACK) ) {//double-tap back-jump does backflip vec3_t fwd, fwdAngles = {0, pm->ps->viewangles[YAW], 0}; AngleVectors( fwdAngles, fwd, NULL, NULL ); pm->ps->velocity[0] = pm->ps->velocity[1] = 0; VectorMA( pm->ps->velocity, -150, fwd, pm->ps->velocity ); //pm->ps->velocity[2] = JUMP_VELOCITY; int parts = SETANIM_LEGS; if ( !pm->ps->weaponTime ) { parts = SETANIM_BOTH; } PM_SetAnim( pm, parts, PM_PickAnim( pm->gent, BOTH_FLIP_BACK1, BOTH_FLIP_BACK3 ), SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD, 0 ); pm->ps->forceJumpZStart = pm->ps->origin[2];//so we don't take damage if we land at same height pm->ps->pm_flags |= PMF_JUMPING|PMF_SLOW_MO_FALL; pm->cmd.upmove = 0; G_SoundOnEnt( pm->gent, CHAN_BODY, "sound/weapons/force/jump.wav" ); WP_ForcePowerDrain( pm->gent, FP_LEVITATION, 0 ); } */ /* else if ( pm->cmd.forwardmove > 0 //pushing forward && pm->ps->forceRageRecoveryTime < pm->cmd.serverTime //not in a force Rage recovery period && pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_1 //have force jump 2 or higher && (level.time - pm->ps->lastOnGround) <= 250//just jumped && (pm->ps->legsAnim == BOTH_JUMP1 || pm->ps->legsAnim == BOTH_INAIR1 )//not in a flip or spin or anything {//run up wall, flip backwards //FIXME: have to be moving... make sure it's opposite the wall... or at least forward? int wallWalkAnim = BOTH_WALL_FLIP_BACK1; int parts = SETANIM_LEGS; int contents = CONTENTS_SOLID; qboolean kick = qtrue; if ( pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_2 ) { wallWalkAnim = BOTH_FORCEWALLRUNFLIP_START; parts = SETANIM_BOTH; kick = qfalse; } else { contents |= CONTENTS_BODY; if ( !pm->ps->weaponTime ) { parts = SETANIM_BOTH; } } if ( PM_HasAnimation( pm->gent, wallWalkAnim ) ) { vec3_t fwd, traceto, mins = {pm->mins[0],pm->mins[1],0}, maxs = {pm->maxs[0],pm->maxs[1],24}, fwdAngles = {0, pm->ps->viewangles[YAW], 0}; trace_t trace; vec3_t idealNormal; AngleVectors( fwdAngles, fwd, NULL, NULL ); VectorMA( pm->ps->origin, 32, fwd, traceto ); pm->trace( &trace, pm->ps->origin, mins, maxs, traceto, pm->ps->clientNum, contents );//FIXME: clip brushes too? VectorSubtract( pm->ps->origin, traceto, idealNormal ); VectorNormalize( idealNormal ); gentity_t *traceEnt = &g_entities[trace.entityNum]; if ( trace.fraction < 1.0f &&((trace.entityNums.solid!=SOLID_BMODEL)||DotProduct(trace.plane.normal,idealNormal)>0.7) ) {//there is a wall there pm->ps->velocity[0] = pm->ps->velocity[1] = 0; if ( wallWalkAnim == BOTH_FORCEWALLRUNFLIP_START ) { pm->ps->velocity[2] = forceJumpStrength[FORCE_LEVEL_3]/2.0f; } else { VectorMA( pm->ps->velocity, -150, fwd, pm->ps->velocity ); } //animate me PM_SetAnim( pm, parts, wallWalkAnim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD, 0 ); pm->ps->forceJumpZStart = pm->ps->origin[2];//so we don't take damage if we land at same height pm->ps->pm_flags |= PMF_JUMPING|PMF_SLOW_MO_FALL; pm->cmd.upmove = 0; G_SoundOnEnt( pm->gent, CHAN_BODY, "sound/weapons/force/jump.wav" ); WP_ForcePowerDrain( pm->gent, FP_LEVITATION, 0 ); //kick if jumping off an ent if ( kick && pm->gent && trace.entityNum < ENTITYNUM_WORLD ) { if ( traceEnt && traceEnt->client && traceEnt->health > 0 && traceEnt->takedamage && traceEnt->client->NPC_class != CLASS_GALAKMECH && traceEnt->client->NPC_class != CLASS_DESANN && !(traceEnt->flags&FL_NO_KNOCKBACK) ) {//push them away and do pain vec3_t oppDir; float strength = VectorNormalize2( pm->ps->velocity, oppDir ); VectorScale( oppDir, -1, oppDir ); //FIXME: need knockdown anim G_Damage( traceEnt, pm->gent, pm->gent, oppDir, traceEnt->currentOrigin, 10, DAMAGE_NO_ARMOR|DAMAGE_NO_HIT_LOC|DAMAGE_NO_KNOCKBACK, MOD_MELEE ); G_Sound( traceEnt, G_SoundIndex( va("sound/weapons/melee/punch%d", Q_irand(1, 4)) ) ); if ( trace.fraction <= 0.5f ) {//close to him if ( traceEnt->health > 0 ) {//didn't kill him if ( (traceEnt->s.number==0&&!Q_irand(0,g_spskill->integer)) || (traceEnt->NPC!=NULL&&Q_irand(RANK_CIVILIAN,traceEnt->NPC->rank)+Q_irand(-2,2)ps->legsAnimTimer<=100 ||!PM_InSpecialJump( legsAnim )//not in a special jump anim ||PM_InReboundJump( legsAnim )//we're already in a rebound ||PM_InBackFlip( legsAnim ) )//a backflip (needed so you can jump off a wall behind you) //&& pm->ps->velocity[2] <= 0 && pm->ps->velocity[2] > -1200 //not falling down very fast && !(pm->ps->pm_flags&PMF_JUMP_HELD)//have to have released jump since last press && (pm->cmd.forwardmove||pm->cmd.rightmove)//pushing in a direction //&& pm->ps->forceRageRecoveryTime < pm->cmd.serverTime //not in a force Rage recovery period && pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_2//level 3 jump or better && pm->ps->forcePower > 10 //have enough force power to do another one && (level.time-pm->ps->lastOnGround) > 250 //haven't been on the ground in the last 1/4 of a second && (!(pm->ps->pm_flags&PMF_JUMPING)//not jumping || ( (level.time-pm->ps->lastOnGround) > 250 //we are jumping, but have been in the air for at least half a second &&( g_debugMelee->integer//if you know kung fu, no height cap on wall-grab-jumps || ((pm->ps->origin[2]-pm->ps->forceJumpZStart) < (forceJumpHeightMax[FORCE_LEVEL_3]-(G_ForceWallJumpStrength()/2.0f))) )//can fit at least one more wall jump in (yes, using "magic numbers"... for now) ) ) //&& (pm->ps->legsAnim == BOTH_JUMP1 || pm->ps->legsAnim == BOTH_INAIR1 ) )//not in a flip or spin or anything ) {//see if we're pushing at a wall and jump off it if so if ( !(pm->ps->saber[0].saberFlags&SFL_NO_WALL_GRAB) && ( !pm->ps->dualSabers || !(pm->ps->saber[1].saberFlags&SFL_NO_WALL_GRAB) ) ) {//okay to do wall-grabs with this saber //FIXME: make sure we have enough force power //FIXME: check to see if we can go any higher //FIXME: limit to a certain number of these in a row? //FIXME: maybe don't require a ucmd direction, just check all 4? //FIXME: should stick to the wall for a second, then push off... vec3_t checkDir, traceto, mins = {pm->mins[0],pm->mins[1],0}, maxs = {pm->maxs[0],pm->maxs[1],24}, fwdAngles = {0, pm->ps->viewangles[YAW], 0}; trace_t trace; vec3_t idealNormal; int anim = -1; if ( pm->cmd.rightmove ) { if ( pm->cmd.rightmove > 0 ) { anim = BOTH_FORCEWALLREBOUND_RIGHT; AngleVectors( fwdAngles, NULL, checkDir, NULL ); } else if ( pm->cmd.rightmove < 0 ) { anim = BOTH_FORCEWALLREBOUND_LEFT; AngleVectors( fwdAngles, NULL, checkDir, NULL ); VectorScale( checkDir, -1, checkDir ); } } else if ( pm->cmd.forwardmove > 0 ) { anim = BOTH_FORCEWALLREBOUND_FORWARD; AngleVectors( fwdAngles, checkDir, NULL, NULL ); } else if ( pm->cmd.forwardmove < 0 ) { anim = BOTH_FORCEWALLREBOUND_BACK; AngleVectors( fwdAngles, checkDir, NULL, NULL ); VectorScale( checkDir, -1, checkDir ); } if ( anim != -1 ) {//trace in the dir we're pushing in and see if there's a vertical wall there VectorMA( pm->ps->origin, 16, checkDir, traceto );//was 8 pm->trace( &trace, pm->ps->origin, mins, maxs, traceto, pm->ps->clientNum, CONTENTS_SOLID, (EG2_Collision)0, 0 );//FIXME: clip brushes too? VectorSubtract( pm->ps->origin, traceto, idealNormal ); VectorNormalize( idealNormal ); gentity_t *traceEnt = &g_entities[trace.entityNum]; if ( trace.fraction < 1.0f && fabs(trace.plane.normal[2]) <= MAX_WALL_GRAB_SLOPE &&((trace.entityNums.solid!=SOLID_BMODEL)||DotProduct(trace.plane.normal,idealNormal)>0.7) ) {//there is a wall there float dot = DotProduct( pm->ps->velocity, trace.plane.normal ); if ( dot < 1.0f ) {//can't be heading *away* from the wall! //grab it! PM_GrabWallForJump( anim ); } } } } } else { //FIXME: if in a butterfly, kick people away? } } } if ( pm->gent //&& pm->cmd.upmove > 0 && pm->ps->forceRageRecoveryTime < pm->cmd.serverTime //not in a force Rage recovery period && pm->ps->weapon == WP_SABER && (pm->ps->weaponTime > 0||(pm->cmd.buttons&BUTTON_ATTACK)) && ((pm->ps->clientNum&&!PM_ControlledByPlayer())||((pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && cg.renderingThirdPerson && !cg.zoomMode)) ) {//okay, we just jumped and we're in an attack if ( !PM_RollingAnim( pm->ps->legsAnim ) && !PM_InKnockDown( pm->ps ) && !PM_InDeathAnim() && !PM_PainAnim( pm->ps->torsoAnim ) && !PM_FlippingAnim( pm->ps->legsAnim ) && !PM_SpinningAnim( pm->ps->legsAnim ) && !PM_SaberInSpecialAttack( pm->ps->torsoAnim ) ) {//HMM... do NPCs need this logic? if ( !PM_SaberInTransitionAny( pm->ps->saberMove ) //not going to/from/between an attack anim && !PM_SaberInAttack( pm->ps->saberMove ) //not in attack anim && pm->ps->weaponTime <= 0//not busy && (pm->cmd.buttons&BUTTON_ATTACK) )//want to attack {//not in an attack or finishing/starting/transitioning one if ( PM_CheckBackflipAttackMove() ) { PM_SetSaberMove( PM_SaberBackflipAttackMove() );//backflip attack } /* else if ( PM_CheckSaberDualJumpAttackMove() ) { PM_SetSaberMove( PM_SaberDualJumpAttackMove() );//jump forward sideways flip attack } */ } /* else if ( ( PM_SaberInTransitionAny( pm->ps->saberMove ) || PM_SaberInAttack( pm->ps->saberMove ) ) && PM_InAnimForSaberMove( pm->ps->torsoAnim, pm->ps->saberMove ) ) {//not in an anim we shouldn't interrupt //see if it's not too late to start a special jump-attack float animLength = PM_AnimLength( g_entities[pm->ps->clientNum].client->clientInfo.animFileIndex, (animNumber_t)pm->ps->torsoAnim ); if ( animLength - pm->ps->torsoAnimTimer < 500 ) {//just started the saberMove //check for special-case jump attacks if ( PM_CheckFlipOverAttackMove( qtrue ) ) { PM_SetSaberMove( PM_SaberFlipOverAttackMove() ); } else if ( PM_CheckJumpAttackMove() ) { PM_SetSaberMove( PM_SaberJumpAttackMove() ); } } } */ } } if ( pm->ps->groundEntityNum == ENTITYNUM_NONE ) { return qfalse; } if ( pm->cmd.upmove > 0 ) {//no special jumps /* gentity_t *groundEnt = &g_entities[pm->ps->groundEntityNum]; if ( groundEnt && groundEnt->NPC ) {//Can't jump off of someone's head return qfalse; } */ pm->ps->velocity[2] = JUMP_VELOCITY; pm->ps->forceJumpZStart = pm->ps->origin[2];//so we don't take damage if we land at same height pm->ps->pm_flags |= PMF_JUMPING; } if ( d_JediAI->integer ) { if ( pm->ps->clientNum && pm->ps->weapon == WP_SABER ) { Com_Printf( "jumping\n" ); } } //Jumping pml.groundPlane = qfalse; pml.walking = qfalse; pm->ps->pm_flags |= PMF_JUMP_HELD; pm->ps->groundEntityNum = ENTITYNUM_NONE; pm->ps->jumpZStart = pm->ps->origin[2]; if ( pm->gent ) { if ( !Q3_TaskIDPending( pm->gent, TID_CHAN_VOICE ) ) { PM_AddEvent( EV_JUMP ); } } else { PM_AddEvent( EV_JUMP ); } //Set the animations if ( pm->ps->gravity > 0 && !PM_InSpecialJump( pm->ps->legsAnim ) && !PM_InGetUp( pm->ps ) ) { PM_JumpForDir(); } return qtrue; } /* ============= PM_CheckWaterJump ============= */ static qboolean PM_CheckWaterJump( void ) { vec3_t spot; int cont; vec3_t flatforward; if (pm->ps->pm_time) { return qfalse; } if ( pm->cmd.forwardmove <= 0 && pm->cmd.upmove <= 0 ) {//they must not want to get out? return qfalse; } // check for water jump if ( pm->waterlevel != 2 ) { return qfalse; } if ( pm->watertype & CONTENTS_LADDER ) { if (pm->ps->velocity[2] <= 0) return qfalse; } flatforward[0] = pml.forward[0]; flatforward[1] = pml.forward[1]; flatforward[2] = 0; VectorNormalize( flatforward ); VectorMA( pm->ps->origin, 30, flatforward, spot ); spot[2] += 24; cont = pm->pointcontents (spot, pm->ps->clientNum ); if ( !(cont & CONTENTS_SOLID) ) { return qfalse; } spot[2] += 16; cont = pm->pointcontents( spot, pm->ps->clientNum ); if ( cont&(CONTENTS_SOLID|CONTENTS_PLAYERCLIP|CONTENTS_WATER|CONTENTS_SLIME|CONTENTS_LAVA|CONTENTS_BODY) ) { return qfalse; } // jump out of water VectorScale( pml.forward, 200, pm->ps->velocity ); pm->ps->velocity[2] = 350+((pm->ps->waterheight-pm->ps->origin[2])*2); pm->ps->pm_flags |= PMF_TIME_WATERJUMP; pm->ps->pm_time = 2000; return qtrue; } //============================================================================ /* =================== PM_WaterJumpMove Flying out of the water =================== */ static void PM_WaterJumpMove( void ) { // waterjump has no control, but falls PM_StepSlideMove( 1 ); pm->ps->velocity[2] -= pm->ps->gravity * pml.frametime; if (pm->ps->velocity[2] < 0) { // cancel as soon as we are falling down again pm->ps->pm_flags &= ~PMF_ALL_TIMES; pm->ps->pm_time = 0; } } /* =================== PM_WaterMove =================== */ static void PM_WaterMove( void ) { int i; vec3_t wishvel; float wishspeed; vec3_t wishdir; float scale; float vel; if ( PM_CheckWaterJump() ) { PM_WaterJumpMove(); return; } else if ( pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_0 && pm->waterlevel < 3 ) { if ( PM_CheckJump () ) { // jumped away return; } } #if 0 // jump = head for surface if ( pm->cmd.upmove >= 10 ) { if (pm->ps->velocity[2] > -300) { if ( pm->watertype == CONTENTS_WATER ) { pm->ps->velocity[2] = 100; } else if (pm->watertype == CONTENTS_SLIME) { pm->ps->velocity[2] = 80; } else { pm->ps->velocity[2] = 50; } } } #endif PM_Friction (); scale = PM_CmdScale( &pm->cmd ); // // user intentions // if ( !scale ) { wishvel[0] = 0; wishvel[1] = 0; if ( pm->watertype & CONTENTS_LADDER ) { wishvel[2] = 0; } else { wishvel[2] = -60; // sink towards bottom } } else { for (i=0 ; i<3 ; i++) { wishvel[i] = scale * pml.forward[i]*pm->cmd.forwardmove + scale * pml.right[i]*pm->cmd.rightmove; } wishvel[2] += scale * pm->cmd.upmove; if ( !(pm->watertype&CONTENTS_LADDER) ) //ladder { float depth = (pm->ps->origin[2]+pm->gent->client->standheight)-pm->ps->waterheight; if ( depth >= 12 ) {//too high! wishvel[2] -= 120; // sink towards bottom if ( wishvel[2] > 0 ) { wishvel[2] = 0; } } else if ( pm->ps->waterHeightLevel >= WHL_UNDER )//!depth && pm->waterlevel == 3 ) { } else if ( depth < 12 ) {//still deep wishvel[2] -= 60; // sink towards bottom if ( wishvel[2] > 30 ) { wishvel[2] = 30; } } } } VectorCopy (wishvel, wishdir); wishspeed = VectorNormalize(wishdir); if ( pm->watertype & CONTENTS_LADDER ) //ladder { if ( wishspeed > pm->ps->speed * pm_ladderScale ) { wishspeed = pm->ps->speed * pm_ladderScale; } PM_Accelerate( wishdir, wishspeed, pm_flyaccelerate ); } else { if ( pm->ps->gravity < 0 ) {//float up pm->ps->velocity[2] -= pm->ps->gravity * pml.frametime; } if ( wishspeed > pm->ps->speed * pm_swimScale ) { wishspeed = pm->ps->speed * pm_swimScale; } PM_Accelerate( wishdir, wishspeed, pm_wateraccelerate ); } // make sure we can go up slopes easily under water if ( pml.groundPlane && DotProduct( pm->ps->velocity, pml.groundTrace.plane.normal ) < 0 ) { vel = VectorLength(pm->ps->velocity); // slide along the ground plane PM_ClipVelocity (pm->ps->velocity, pml.groundTrace.plane.normal, pm->ps->velocity, OVERCLIP ); VectorNormalize(pm->ps->velocity); VectorScale(pm->ps->velocity, vel, pm->ps->velocity); } PM_SlideMove( qfalse ); } /* =================== PM_FlyVehicleMove =================== */ static void PM_FlyVehicleMove( void ) { int i; vec3_t wishvel; float wishspeed; vec3_t wishdir; float scale; float zVel; float fmove = 0.0f, smove = 0.0f; // We don't use these here because we pre-calculate the movedir in the vehicle update anyways, and if // you leave this, you get strange motion during boarding (the player can move the vehicle). //fmove = pm->cmd.forwardmove; //smove = pm->cmd.rightmove; // normal slowdown if ( pm->ps->gravity && pm->ps->velocity[2] < 0 && pm->ps->groundEntityNum == ENTITYNUM_NONE ) {//falling zVel = pm->ps->velocity[2]; PM_Friction (); pm->ps->velocity[2] = zVel; } else { PM_Friction (); if ( pm->ps->velocity[2] < 0 && pm->ps->groundEntityNum != ENTITYNUM_NONE ) { pm->ps->velocity[2] = 0; // ignore slope movement } } scale = PM_CmdScale( &pm->cmd ); // Get The WishVel And WishSpeed //------------------------------- if ( pm->ps->clientNum && (USENEWNAVSYSTEM || !VectorCompare( pm->ps->moveDir, vec3_origin )) ) {//NPC // If The UCmds Were Set, But Never Converted Into A MoveDir, Then Make The WishDir From UCmds //-------------------------------------------------------------------------------------------- if ((fmove!=0.0f || smove!=0.0f) && VectorCompare(pm->ps->moveDir, vec3_origin)) { //gi.Printf("Generating MoveDir\n"); for ( i = 0 ; i < 3 ; i++ ) { wishvel[i] = pml.forward[i]*fmove + pml.right[i]*smove; } VectorCopy( wishvel, wishdir ); wishspeed = VectorNormalize(wishdir); wishspeed *= scale; } // Otherwise, Use The Move Dir //----------------------------- else { wishspeed = pm->ps->speed; VectorScale( pm->ps->moveDir, pm->ps->speed, wishvel ); VectorCopy( pm->ps->moveDir, wishdir ); } } else { for ( i = 0 ; i < 3 ; i++ ) { wishvel[i] = pml.forward[i]*fmove + pml.right[i]*smove; } // when going up or down slopes the wish velocity should Not be zero // wishvel[2] = 0; VectorCopy (wishvel, wishdir); wishspeed = VectorNormalize(wishdir); wishspeed *= scale; } // Handle negative speed. if ( wishspeed < 0 ) { wishspeed = wishspeed * -1.0f; VectorScale( wishvel, -1.0f, wishvel ); VectorScale( wishdir, -1.0f, wishdir ); } VectorCopy( wishvel, wishdir ); wishspeed = VectorNormalize( wishdir ); PM_Accelerate( wishdir, wishspeed, 100 ); PM_StepSlideMove( 1 ); } /* =================== PM_FlyMove Only with the flight powerup =================== */ static void PM_FlyMove( void ) { int i; vec3_t wishvel; float wishspeed; vec3_t wishdir; float scale; float accel; qboolean lowGravMove = qfalse; qboolean jetPackMove = qfalse; // normal slowdown PM_Friction (); if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && pm->gent && pm->gent->client && (pm->gent->client->NPC_class == CLASS_BOBAFETT||pm->gent->client->NPC_class == CLASS_ROCKETTROOPER) && pm->gent->client->moveType == MT_FLYSWIM ) {//jetpack accel accel = pm_flyaccelerate; jetPackMove = qtrue; } else if ( pm->ps->gravity <= 0 && ((pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) || (pm->gent&&pm->gent->client&&pm->gent->client->moveType == MT_RUNJUMP)) ) { PM_CheckJump(); accel = 1.0f; pm->ps->velocity[2] -= pm->ps->gravity * pml.frametime; pm->ps->jumpZStart = pm->ps->origin[2];//so we don't take a lot of damage when the gravity comes back on lowGravMove = qtrue; } else { accel = pm_flyaccelerate; } scale = PM_CmdScale( &pm->cmd ); // // user intentions // if ( !scale ) { wishvel[0] = 0; wishvel[1] = 0; wishvel[2] = 0; } else { for (i=0 ; i<3 ; i++) { wishvel[i] = scale * pml.forward[i]*pm->cmd.forwardmove + scale * pml.right[i]*pm->cmd.rightmove; } if ( jetPackMove ) { wishvel[2] += pm->cmd.upmove; } else if ( lowGravMove ) { wishvel[2] += scale * pm->cmd.upmove; VectorScale( wishvel, 0.5f, wishvel ); } } VectorCopy( wishvel, wishdir ); wishspeed = VectorNormalize( wishdir ); PM_Accelerate( wishdir, wishspeed, accel ); PM_StepSlideMove( 1 ); } qboolean PM_GroundSlideOkay( float zNormal ) { if ( zNormal > 0 ) { if ( pm->ps->velocity[2] > 0 ) { if ( pm->ps->legsAnim == BOTH_WALL_RUN_RIGHT || pm->ps->legsAnim == BOTH_WALL_RUN_LEFT || pm->ps->legsAnim == BOTH_WALL_RUN_RIGHT_STOP || pm->ps->legsAnim == BOTH_WALL_RUN_LEFT_STOP || pm->ps->legsAnim == BOTH_FORCEWALLRUNFLIP_START || pm->ps->legsAnim == BOTH_FORCELONGLEAP_START || pm->ps->legsAnim == BOTH_FORCELONGLEAP_ATTACK || pm->ps->legsAnim == BOTH_FORCELONGLEAP_LAND || PM_InReboundJump( pm->ps->legsAnim )) { return qfalse; } } } return qtrue; } /* =================== PM_AirMove =================== */ static void PM_AirMove( void ) { int i; vec3_t wishvel; float fmove, smove; vec3_t wishdir; float wishspeed; usercmd_t cmd; float gravMod = 1.0f; #if METROID_JUMP PM_CheckJump(); #endif PM_Friction(); fmove = pm->cmd.forwardmove; smove = pm->cmd.rightmove; cmd = pm->cmd; PM_CmdScale( &cmd ); // set the movementDir so clients can rotate the legs for strafing PM_SetMovementDir(); // project moves down to flat plane pml.forward[2] = 0; pml.right[2] = 0; VectorNormalize (pml.forward); VectorNormalize (pml.right); Vehicle_t *pVeh = NULL; if ( pm->gent->client && pm->gent->client->NPC_class == CLASS_VEHICLE ) { pVeh = pm->gent->m_pVehicle; } if ( pVeh && pVeh->m_pVehicleInfo->hoverHeight > 0 ) {//in a hovering vehicle, have air control // Flying Or Breaking, No Control //-------------------------------- if ( pVeh->m_ulFlags&VEH_FLYING || pVeh->m_ulFlags&VEH_SLIDEBREAKING) { wishspeed = 0.0f; VectorClear( wishvel ); VectorClear( wishdir ); } // Out Of Control - Maintain pos3 Velocity //----------------------------------------- else if ((pVeh->m_ulFlags&VEH_OUTOFCONTROL) || (pVeh->m_ulFlags&VEH_STRAFERAM)) { VectorCopy(pm->gent->pos3, wishvel); VectorCopy(wishvel, wishdir); wishspeed = VectorNormalize(wishdir); } // Boarding - Maintain Boarding Velocity //--------------------------------------- else if (pVeh->m_iBoarding) { VectorCopy(pVeh->m_vBoardingVelocity, wishvel); VectorCopy(wishvel, wishdir); wishspeed = VectorNormalize(wishdir); } // Otherwise, Normal Velocity //---------------------------- else { wishspeed = pm->ps->speed; VectorScale( pm->ps->moveDir, pm->ps->speed, wishvel ); VectorCopy( pm->ps->moveDir, wishdir ); } } else if ( (pm->ps->pm_flags&PMF_SLOW_MO_FALL) ) {//no air-control VectorClear( wishvel ); } else { for ( i = 0 ; i < 2 ; i++ ) { wishvel[i] = pml.forward[i]*fmove + pml.right[i]*smove; } wishvel[2] = 0; } VectorCopy (wishvel, wishdir); wishspeed = VectorNormalize(wishdir); if ( ( DotProduct (pm->ps->velocity, wishdir) ) < 0.0f ) {//Encourage deceleration away from the current velocity wishspeed *= pm_airDecelRate; } // not on ground, so little effect on velocity float accelerate = pm_airaccelerate; if ( pVeh && pVeh->m_pVehicleInfo->type == VH_SPEEDER ) {//speeders have more control in air //in mid-air accelerate = pVeh->m_pVehicleInfo->acceleration; if ( pml.groundPlane ) {//on a slope of some kind, shouldn't have much control and should slide a lot accelerate *= 0.5f; } if (pVeh->m_ulFlags & VEH_SLIDEBREAKING) { VectorScale(pm->ps->velocity, 0.80f, pm->ps->velocity); } if (pm->ps->velocity[2]>1000.0f) { pm->ps->velocity[2] = 1000.0f; } } PM_Accelerate( wishdir, wishspeed, accelerate ); // we may have a ground plane that is very steep, even // though we don't have a groundentity // slide along the steep plane if ( pml.groundPlane ) { if ( PM_GroundSlideOkay( pml.groundTrace.plane.normal[2] ) ) { PM_ClipVelocity( pm->ps->velocity, pml.groundTrace.plane.normal, pm->ps->velocity, OVERCLIP ); } } if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_0 && pm->ps->forceJumpZStart && pm->ps->velocity[2] > 0 ) {//I am force jumping and I'm not holding the button anymore float curHeight = pm->ps->origin[2] - pm->ps->forceJumpZStart + (pm->ps->velocity[2]*pml.frametime); float maxJumpHeight = forceJumpHeight[pm->ps->forcePowerLevel[FP_LEVITATION]]; if ( curHeight >= maxJumpHeight ) {//reached top, cut velocity pm->ps->velocity[2] = 0; } } if ( (pm->ps->pm_flags&PMF_STUCK_TO_WALL) ) { gravMod = 0.0f; } PM_StepSlideMove( gravMod ); if (pVeh && pm->ps->pm_flags&PMF_BUMPED) { /* // Turn Vehicle In Direction Of Collision //---------------------------------------- vec3_t nAngles; vectoangles(pm->ps->velocity, nAngles); nAngles[0] = pVeh->m_pParentEntity->client->ps.viewangles[0]; nAngles[2] = pVeh->m_pParentEntity->client->ps.viewangles[2]; // toggle the teleport bit so the client knows to not lerp player->client->ps.eFlags ^= EF_TELEPORT_BIT; // set angles SetClientViewAngle( pVeh->m_pParentEntity, nAngles ); if (pVeh->m_pPilot) { SetClientViewAngle( pVeh->m_pPilot, nAngles ); } VectorCopy(nAngles, pVeh->m_vPrevOrientation); VectorCopy(nAngles, pVeh->m_vOrientation); pVeh->m_vAngularVelocity = 0.0f; */ // Reduce "Bounce Up Wall" Velocity //---------------------------------- if (pm->ps->velocity[2]>0) { pm->ps->velocity[2] *= 0.1f; } } } /* =================== PM_WalkMove =================== */ static void PM_WalkMove( void ) { int i; vec3_t wishvel; float fmove, smove; vec3_t wishdir; float wishspeed; float scale; usercmd_t cmd; float accelerate; float vel; if ( pm->ps->gravity < 0 ) {//float away pm->ps->velocity[2] -= pm->ps->gravity * pml.frametime; pm->ps->groundEntityNum = ENTITYNUM_NONE; pml.groundPlane = qfalse; pml.walking = qfalse; if ( pm->waterlevel > 1 ) { PM_WaterMove(); } else { PM_AirMove(); } return; } if ( pm->waterlevel > 2 && DotProduct( pml.forward, pml.groundTrace.plane.normal ) > 0 ) { // begin swimming PM_WaterMove(); return; } if ( PM_CheckJump () ) { // jumped away if ( pm->waterlevel > 1 ) { PM_WaterMove(); } else { PM_AirMove(); } return; } if ( pm->ps->groundEntityNum != ENTITYNUM_NONE &&//on ground pm->ps->velocity[2] <= 0 &&//not going up pm->ps->pm_flags&PMF_TIME_KNOCKBACK )//knockback fimter on (stops friction) { pm->ps->pm_flags &= ~PMF_TIME_KNOCKBACK; } qboolean slide = qfalse; if ( pm->ps->pm_type == PM_DEAD ) {//corpse if ( g_entities[pm->ps->groundEntityNum].client ) {//on a client if ( g_entities[pm->ps->groundEntityNum].health > 0 ) {//a living client //no friction slide = qtrue; } } } if ( !slide ) { PM_Friction (); } if ( g_debugMelee->integer ) { if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer())//player && cg.renderingThirdPerson//in third person && ((pm->cmd.buttons&BUTTON_USE)||pm->ps->leanStopDebounceTime)//holding use or leaning //&& (pm->ps->forcePowersActive&(1<ps->groundEntityNum != ENTITYNUM_NONE//on ground && !cg_usingInFrontOf )//nothing to use {//holding use stops you from moving return; } } fmove = pm->cmd.forwardmove; smove = pm->cmd.rightmove; cmd = pm->cmd; scale = PM_CmdScale( &cmd ); // set the movementDir so clients can rotate the legs for strafing PM_SetMovementDir(); // project moves down to flat plane pml.forward[2] = 0; pml.right[2] = 0; // project the forward and right directions onto the ground plane PM_ClipVelocity (pml.forward, pml.groundTrace.plane.normal, pml.forward, OVERCLIP ); PM_ClipVelocity (pml.right, pml.groundTrace.plane.normal, pml.right, OVERCLIP ); // VectorNormalize (pml.forward); VectorNormalize (pml.right); /* if ( ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_VEHICLE ) && pm->gent->NPC ) {//speeder control scheme vec3_t vfwd, vrt; AngleVectors( ((CVehicleNPC *)pm->gent->NPC)->m_vOrientation, vfwd, vrt, NULL ); float speed = pm->ps->speed; if ( fmove < 0 ) {//going backwards if ( speed < 0 ) {//speed is negative, but since our command is reverse, make speed positive speed = fabs( speed ); } else if ( speed > 0 ) {//trying to move back but speed is still positive, so keep moving forward (we'll slow down eventually) speed *= -1; } } VectorScale( vfwd, speed*fmove/127.0f, wishvel ); //VectorMA( wishvel, pm->ps->speed*smove/127.0f, vrt, wishvel ); wishspeed = VectorNormalize2( wishvel, wishdir ); } else*/ // Get The WishVel And WishSpeed //------------------------------- if ( pm->ps->clientNum && (USENEWNAVSYSTEM || !VectorCompare( pm->ps->moveDir, vec3_origin )) ) {//NPC // If The UCmds Were Set, But Never Converted Into A MoveDir, Then Make The WishDir From UCmds //-------------------------------------------------------------------------------------------- if ((fmove!=0.0f || smove!=0.0f) && VectorCompare(pm->ps->moveDir, vec3_origin)) { //gi.Printf("Generating MoveDir\n"); for ( i = 0 ; i < 3 ; i++ ) { wishvel[i] = pml.forward[i]*fmove + pml.right[i]*smove; } VectorCopy( wishvel, wishdir ); wishspeed = VectorNormalize(wishdir); wishspeed *= scale; } // Otherwise, Use The Move Dir //----------------------------- else { wishspeed = pm->ps->speed; VectorScale( pm->ps->moveDir, pm->ps->speed, wishvel ); VectorCopy( pm->ps->moveDir, wishdir ); } } else { for ( i = 0 ; i < 3 ; i++ ) { wishvel[i] = pml.forward[i]*fmove + pml.right[i]*smove; } // when going up or down slopes the wish velocity should Not be zero // wishvel[2] = 0; VectorCopy (wishvel, wishdir); wishspeed = VectorNormalize(wishdir); wishspeed *= scale; } // Handle negative speed. if ( wishspeed < 0 ) { wishspeed = wishspeed * -1.0f; VectorScale( wishvel, -1.0f, wishvel ); VectorScale( wishdir, -1.0f, wishdir ); } // clamp the speed lower if ducking if ( pm->ps->pm_flags & PMF_DUCKED && !PM_InKnockDown( pm->ps ) ) { if ( wishspeed > pm->ps->speed * pm_duckScale ) { wishspeed = pm->ps->speed * pm_duckScale; } } // clamp the speed lower if wading or walking on the bottom if ( pm->waterlevel ) { float waterScale; waterScale = pm->waterlevel / 3.0; waterScale = 1.0 - ( 1.0 - pm_swimScale ) * waterScale; if ( wishspeed > pm->ps->speed * waterScale ) { wishspeed = pm->ps->speed * waterScale; } } // when a player gets hit, they temporarily lose // full control, which allows them to be moved a bit if ( Flying == FLY_HOVER ) { accelerate = pm_vehicleaccelerate; } else if ( ( pml.groundTrace.surfaceFlags & SURF_SLICK ) || (pm->ps->pm_flags&PMF_TIME_KNOCKBACK) || (pm->ps->pm_flags&PMF_TIME_NOFRICTION) ) { accelerate = pm_airaccelerate; } else { accelerate = pm_accelerate; // Wind Affects Acceleration //=================================================== if (wishspeed>0.0f && pm->gent && !pml.walking) { if (gi.WE_GetWindGusting(pm->gent->currentOrigin)) { vec3_t windDir; if (gi.WE_GetWindVector(windDir, pm->gent->currentOrigin)) { if (gi.WE_IsOutside(pm->gent->currentOrigin)) { VectorScale(windDir, -1.0f, windDir); accelerate *= (1.0f - (DotProduct(wishdir, windDir)*0.55f)); } } } } //=================================================== } PM_Accelerate (wishdir, wishspeed, accelerate); //Com_Printf("velocity = %1.1f %1.1f %1.1f\n", pm->ps->velocity[0], pm->ps->velocity[1], pm->ps->velocity[2]); //Com_Printf("velocity1 = %1.1f\n", VectorLength(pm->ps->velocity)); if ( ( pml.groundTrace.surfaceFlags & SURF_SLICK ) || pm->ps->pm_flags & PMF_TIME_KNOCKBACK || (pm->ps->pm_flags&PMF_TIME_NOFRICTION) ) { if ( pm->ps->gravity >= 0 && pm->ps->groundEntityNum != ENTITYNUM_NONE && !VectorLengthSquared( pm->ps->velocity ) && pml.groundTrace.plane.normal[2] == 1.0 ) {//on ground and not moving and on level ground, no reason to do stupid fucking gravity with the clipvelocity!!!! } else { if ( !(pm->ps->eFlags&EF_FORCE_GRIPPED) && !(pm->ps->eFlags&EF_FORCE_DRAINED) ) { pm->ps->velocity[2] -= pm->ps->gravity * pml.frametime; } } } else { // don't reset the z velocity for slopes // pm->ps->velocity[2] = 0; } vel = VectorLength(pm->ps->velocity); // slide along the ground plane PM_ClipVelocity (pm->ps->velocity, pml.groundTrace.plane.normal, pm->ps->velocity, OVERCLIP ); // don't decrease velocity when going up or down a slope VectorNormalize(pm->ps->velocity); VectorScale(pm->ps->velocity, vel, pm->ps->velocity); // don't do anything if standing still if ( !pm->ps->velocity[0] && !pm->ps->velocity[1] ) { return; } if ( pm->ps->gravity <= 0 ) {//need to apply gravity since we're going to float up from ground PM_StepSlideMove( 1 ); } else { PM_StepSlideMove( 0 ); } //Com_Printf("velocity2 = %1.1f\n", VectorLength(pm->ps->velocity)); } /* ============== PM_DeadMove ============== */ static void PM_DeadMove( void ) { float forward; // If this is a vehicle, tell him he's dead, but give him a little while to do his things. /* if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_VEHICLE && pm->gent->NPC && pm->gent->health != -99999 ) { pm->gent->health = 1; ((CVehicleNPC *)pm->gent->NPC)->StartDeathDelay( 0 ); } else { pm->gent->health = 0; }*/ if ( !pml.walking ) { return; } // extra friction forward = VectorLength (pm->ps->velocity); forward -= 20; if ( forward <= 0 ) { VectorClear (pm->ps->velocity); } else { VectorNormalize (pm->ps->velocity); VectorScale (pm->ps->velocity, forward, pm->ps->velocity); } } /* =============== PM_NoclipMove =============== */ static void PM_NoclipMove( void ) { float speed, drop, friction, control, newspeed; int i; vec3_t wishvel; float fmove, smove; vec3_t wishdir; float wishspeed; float scale; if(pm->gent && pm->gent->client) { pm->ps->viewheight = pm->gent->client->standheight + STANDARD_VIEWHEIGHT_OFFSET; // if ( !pm->gent->mins[0] || !pm->gent->mins[1] || !pm->gent->mins[2] || !pm->gent->maxs[0] || !pm->gent->maxs[1] || !pm->gent->maxs[2] ) // { // assert(0); // } VectorCopy( pm->gent->mins, pm->mins ); VectorCopy( pm->gent->maxs, pm->maxs ); } else { pm->ps->viewheight = DEFAULT_MAXS_2 + STANDARD_VIEWHEIGHT_OFFSET;//DEFAULT_VIEWHEIGHT; if ( !DEFAULT_MINS_0 || !DEFAULT_MINS_1 || !DEFAULT_MAXS_0 || !DEFAULT_MAXS_1 || !DEFAULT_MINS_2 || !DEFAULT_MAXS_2 ) { assert(0); } pm->mins[0] = DEFAULT_MINS_0; pm->mins[1] = DEFAULT_MINS_1; pm->mins[2] = DEFAULT_MINS_2; pm->maxs[0] = DEFAULT_MAXS_0; pm->maxs[1] = DEFAULT_MAXS_1; pm->maxs[2] = DEFAULT_MAXS_2; } // friction speed = VectorLength (pm->ps->velocity); if (speed < 1) { VectorCopy (vec3_origin, pm->ps->velocity); } else { drop = 0; friction = pm_friction*1.5; // extra friction control = speed < pm_stopspeed ? pm_stopspeed : speed; drop += control*friction*pml.frametime; // scale the velocity newspeed = speed - drop; if (newspeed < 0) newspeed = 0; newspeed /= speed; VectorScale (pm->ps->velocity, newspeed, pm->ps->velocity); } // accelerate scale = PM_CmdScale( &pm->cmd ); if (pm->cmd.buttons & BUTTON_ATTACK) { //turbo boost scale *= 10; } if (pm->cmd.buttons & BUTTON_ALT_ATTACK) { //turbo boost scale *= 10; } fmove = pm->cmd.forwardmove; smove = pm->cmd.rightmove; for (i=0 ; i<3 ; i++) wishvel[i] = pml.forward[i]*fmove + pml.right[i]*smove; wishvel[2] += pm->cmd.upmove; VectorCopy (wishvel, wishdir); wishspeed = VectorNormalize(wishdir); wishspeed *= scale; PM_Accelerate( wishdir, wishspeed, pm_accelerate ); // move VectorMA (pm->ps->origin, pml.frametime, pm->ps->velocity, pm->ps->origin); } //============================================================================ static float PM_DamageForDelta( int delta ) { float damage = delta; if ( pm->gent->NPC ) { if ( pm->ps->weapon == WP_SABER || (pm->gent->client && pm->gent->client->NPC_class == CLASS_REBORN) ) {//FIXME: for now Jedi take no falling damage, but really they should if pushed off? damage = 0; } } else if ( pm->ps->clientNum < MAX_CLIENTS ) { if ( damage < 50 ) { if ( damage > 24 ) { damage = damage - 25; } } else { damage *= 0.5f; } } return damage * 0.5f; } static void PM_CrashLandDamage( int damage ) { if ( pm->gent ) { int dflags = DAMAGE_NO_ARMOR; if ( pm->gent->NPC && (pm->gent->NPC->aiFlags&NPCAI_DIE_ON_IMPACT) ) { damage = 1000; dflags |= DAMAGE_DIE_ON_IMPACT; } else { damage = PM_DamageForDelta( damage ); if ( (pm->gent->flags&FL_NO_IMPACT_DMG) ) return; } if ( damage ) { pm->gent->painDebounceTime = level.time + 200; // no normal pain sound G_Damage( pm->gent, NULL, player, NULL, NULL, damage, dflags, MOD_FALLING ); } } } /* static float PM_CrashLandDelta( vec3_t org, vec3_t prevOrg, vec3_t prev_vel, float grav, int waterlevel ) { float delta; float dist; float vel, acc; float t; float a, b, c, den; // calculate the exact velocity on landing dist = org[2] - prevOrg[2]; vel = prev_vel[2]; acc = -grav; a = acc / 2; b = vel; c = -dist; den = b * b - 4 * a * c; if ( den < 0 ) { return 0; } t = (-b - sqrt( den ) ) / ( 2 * a ); delta = vel + t * acc; delta = delta*delta * 0.0001; // never take falling damage if completely underwater if ( waterlevel == 3 ) { return 0; } // reduce falling damage if there is standing water if ( waterlevel == 2 ) { delta *= 0.25; } if ( waterlevel == 1 ) { delta *= 0.5; } return delta; } */ static float PM_CrashLandDelta( vec3_t prev_vel, int waterlevel ) { float delta; if ( pm->waterlevel == 3 ) { return 0; } delta = fabs(prev_vel[2])/10;//VectorLength( prev_vel ) // reduce falling damage if there is standing water if ( pm->waterlevel == 2 ) { delta *= 0.25; } if ( pm->waterlevel == 1 ) { delta *= 0.5; } return delta; } int PM_GetLandingAnim( void ) { int anim = pm->ps->legsAnim; //special cases: if ( anim == BOTH_FLIP_ATTACK7 || anim == BOTH_FLIP_HOLD7 ) { return BOTH_FLIP_LAND; } else if ( anim == BOTH_FLIP_LAND ) { if ( !g_allowBunnyhopping->integer ) { //stick landings some pm->ps->velocity[0] *= 0.5f; pm->ps->velocity[1] *= 0.5f; } return BOTH_LAND1; } else if ( PM_InAirKickingAnim( anim ) ) { switch ( anim ) { case BOTH_A7_KICK_F_AIR: return BOTH_FORCELAND1; break; case BOTH_A7_KICK_B_AIR: return BOTH_FORCELANDBACK1; break; case BOTH_A7_KICK_R_AIR: return BOTH_FORCELANDRIGHT1; break; case BOTH_A7_KICK_L_AIR: return BOTH_FORCELANDLEFT1; break; } } if ( PM_SpinningAnim( anim ) || PM_SaberInSpecialAttack( anim ) ) { return -1; } switch ( anim ) { case BOTH_FORCEJUMPLEFT1: case BOTH_FORCEINAIRLEFT1: anim = BOTH_FORCELANDLEFT1; if ( !g_allowBunnyhopping->integer ) { //stick landings some pm->ps->velocity[0] *= 0.5f; pm->ps->velocity[1] *= 0.5f; } break; case BOTH_FORCEJUMPRIGHT1: case BOTH_FORCEINAIRRIGHT1: anim = BOTH_FORCELANDRIGHT1; if ( !g_allowBunnyhopping->integer ) { //stick landings some pm->ps->velocity[0] *= 0.5f; pm->ps->velocity[1] *= 0.5f; } break; case BOTH_FORCEJUMP1: case BOTH_FORCEINAIR1: if ( !g_allowBunnyhopping->integer ) { //stick landings some pm->ps->velocity[0] *= 0.5f; pm->ps->velocity[1] *= 0.5f; } anim = BOTH_FORCELAND1; break; case BOTH_FORCEJUMPBACK1: case BOTH_FORCEINAIRBACK1: if ( !g_allowBunnyhopping->integer ) { //stick landings some pm->ps->velocity[0] *= 0.5f; pm->ps->velocity[1] *= 0.5f; } anim = BOTH_FORCELANDBACK1; break; case BOTH_JUMPLEFT1: case BOTH_INAIRLEFT1: anim = BOTH_LANDLEFT1; if ( !g_allowBunnyhopping->integer ) { //stick landings some pm->ps->velocity[0] *= 0.5f; pm->ps->velocity[1] *= 0.5f; } break; case BOTH_JUMPRIGHT1: case BOTH_INAIRRIGHT1: anim = BOTH_LANDRIGHT1; if ( !g_allowBunnyhopping->integer ) { //stick landings some pm->ps->velocity[0] *= 0.5f; pm->ps->velocity[1] *= 0.5f; } break; case BOTH_JUMP1: case BOTH_INAIR1: anim = BOTH_LAND1; if ( !g_allowBunnyhopping->integer ) { //stick landings some pm->ps->velocity[0] *= 0.5f; pm->ps->velocity[1] *= 0.5f; } break; case BOTH_JUMPBACK1: case BOTH_INAIRBACK1: anim = BOTH_LANDBACK1; if ( !g_allowBunnyhopping->integer ) { //stick landings some pm->ps->velocity[0] *= 0.5f; pm->ps->velocity[1] *= 0.5f; } break; case BOTH_BUTTERFLY_LEFT: case BOTH_BUTTERFLY_RIGHT: case BOTH_BUTTERFLY_FL1: case BOTH_BUTTERFLY_FR1: case BOTH_FJSS_TR_BL: case BOTH_FJSS_TL_BR: case BOTH_LUNGE2_B__T_: case BOTH_FORCELEAP2_T__B_: case BOTH_ARIAL_LEFT: case BOTH_ARIAL_RIGHT: case BOTH_ARIAL_F1: case BOTH_CARTWHEEL_LEFT: case BOTH_CARTWHEEL_RIGHT: case BOTH_JUMPFLIPSLASHDOWN1://# case BOTH_JUMPFLIPSTABDOWN://# case BOTH_JUMPATTACK6: case BOTH_JUMPATTACK7: case BOTH_A7_KICK_F: case BOTH_A7_KICK_B: case BOTH_A7_KICK_R: case BOTH_A7_KICK_L: case BOTH_A7_KICK_S: case BOTH_A7_KICK_BF: case BOTH_A7_KICK_RL: case BOTH_A7_KICK_F_AIR: case BOTH_A7_KICK_B_AIR: case BOTH_A7_KICK_R_AIR: case BOTH_A7_KICK_L_AIR: case BOTH_STABDOWN: case BOTH_STABDOWN_STAFF: case BOTH_STABDOWN_DUAL: case BOTH_A6_SABERPROTECT: case BOTH_A7_SOULCAL: case BOTH_A1_SPECIAL: case BOTH_A2_SPECIAL: case BOTH_A3_SPECIAL: case BOTH_PULL_IMPALE_STAB: case BOTH_PULL_IMPALE_SWING: anim = -1; break; case BOTH_FORCELONGLEAP_START: case BOTH_FORCELONGLEAP_ATTACK: return BOTH_FORCELONGLEAP_LAND; break; case BOTH_WALL_RUN_LEFT://# case BOTH_WALL_RUN_RIGHT://# if ( pm->ps->legsAnimTimer > 500 ) {//only land at end of anim return -1; } //NOTE: falls through on purpose! default: if ( pm->ps->pm_flags & PMF_BACKWARDS_JUMP ) { anim = BOTH_LANDBACK1; } else { anim = BOTH_LAND1; } if ( !g_allowBunnyhopping->integer ) { //stick landings some pm->ps->velocity[0] *= 0.5f; pm->ps->velocity[1] *= 0.5f; } break; } return anim; } void G_StartRoll( gentity_t *ent, int anim ) { NPC_SetAnim(ent,SETANIM_BOTH,anim,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD|SETANIM_FLAG_HOLDLESS); ent->client->ps.weaponTime = ent->client->ps.torsoAnimTimer - 200;//just to make sure it's cleared when roll is done G_AddEvent( ent, EV_ROLL, 0 ); ent->client->ps.saberMove = LS_NONE; } static qboolean PM_TryRoll( void ) { float rollDist = 192;//was 64; if ( PM_SaberInAttack( pm->ps->saberMove ) || PM_SaberInSpecialAttack( pm->ps->torsoAnim ) || PM_SpinningSaberAnim( pm->ps->legsAnim ) || ((pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer())&&PM_SaberInStart( pm->ps->saberMove )) ) {//attacking or spinning (or, if player, starting an attack) if ( PM_CanRollFromSoulCal( pm->ps ) ) {//hehe } else { return qfalse; } } if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && (!cg.renderingThirdPerson || cg.zoomMode) ) {//player can't do this in 1st person return qfalse; } if ( !pm->gent ) { return qfalse; } if ( (pm->ps->saber[0].saberFlags&SFL_NO_ROLLS) ) { return qfalse; } if ( pm->ps->dualSabers && (pm->ps->saber[1].saberFlags&SFL_NO_ROLLS) ) { return qfalse; } if ( pm->ps->clientNum && pm->gent->NPC ) {//NPC if ( pm->gent->NPC->scriptFlags&SCF_NO_ACROBATICS ) {//scripted to never do acrobatics return qfalse; } if ( pm->ps->weapon == WP_SABER ) {//jedi/reborn if ( pm->gent->NPC->rank != RANK_CREWMAN && pm->gent->NPC->rank < RANK_LT_JG ) {//reborn/jedi who are not acrobats or fencers can't do any of these acrobatics return qfalse; } } else {//non-jedi/reborn if ( pm->ps->weapon != WP_NONE )//not empty-handed...who would that be??? {//only jedi/reborn NPCs should be able to do rolls (with a few exceptions) if ( !pm->gent || !pm->gent->client || (pm->gent->client->NPC_class != CLASS_BOBAFETT //boba can roll with it, baby && pm->gent->client->NPC_class != CLASS_REBORN //reborn using weapons other than saber can still roll )) {//can't roll return qfalse; } } } } vec3_t fwd, right, traceto, mins = { pm->mins[0], pm->mins[1], pm->mins[2] + STEPSIZE }, maxs = { pm->maxs[0], pm->maxs[1], (float)pm->gent->client->crouchheight }, fwdAngles = { 0, pm->ps->viewangles[YAW], 0 }; trace_t trace; int anim = -1; AngleVectors( fwdAngles, fwd, right, NULL ); //FIXME: trace ahead for clearance to roll if ( pm->cmd.forwardmove ) { if ( pm->ps->pm_flags & PMF_BACKWARDS_RUN ) { anim = BOTH_ROLL_B; VectorMA( pm->ps->origin, -rollDist, fwd, traceto ); } else { anim = BOTH_ROLL_F; VectorMA( pm->ps->origin, rollDist, fwd, traceto ); } } else if ( pm->cmd.rightmove > 0 ) { anim = BOTH_ROLL_R; VectorMA( pm->ps->origin, rollDist, right, traceto ); } else if ( pm->cmd.rightmove < 0 ) { anim = BOTH_ROLL_L; VectorMA( pm->ps->origin, -rollDist, right, traceto ); } else {//??? } if ( anim != -1 ) { qboolean roll = qfalse; int clipmask = CONTENTS_SOLID; if ( pm->ps->clientNum ) { clipmask |= (CONTENTS_MONSTERCLIP|CONTENTS_BOTCLIP); } else { if ( pm->gent && pm->gent->enemy && pm->gent->enemy->health > 0 ) {//player can always roll in combat roll = qtrue; } else { clipmask |= CONTENTS_PLAYERCLIP; } } if ( !roll ) { pm->trace( &trace, pm->ps->origin, mins, maxs, traceto, pm->ps->clientNum, clipmask, (EG2_Collision)0, 0 ); if ( trace.fraction >= 1.0f ) {//okay, clear, check for a bottomless drop vec3_t top; VectorCopy( traceto, top ); traceto[2] -= 256; pm->trace( &trace, top, mins, maxs, traceto, pm->ps->clientNum, CONTENTS_SOLID, (EG2_Collision)0, 0 ); if ( trace.fraction < 1.0f ) {//not a bottomless drop roll = qtrue; } } else {//hit an architectural obstruction if ( pm->ps->clientNum ) {//NPCs don't care about rolling into walls, just off ledges if ( !(trace.contents&CONTENTS_BOTCLIP) ) { roll = qtrue; } } else if ( G_EntIsDoor( trace.entityNum ) ) {//okay to roll into a door if ( G_EntIsUnlockedDoor( trace.entityNum ) ) {//if it's an auto-door roll = qtrue; } } else {//check other conditions gentity_t *traceEnt = &g_entities[trace.entityNum]; if ( traceEnt && (traceEnt->svFlags&SVF_GLASS_BRUSH) ) {//okay to roll through glass roll = qtrue; } } } } if ( roll ) { G_StartRoll( pm->gent, anim ); return qtrue; } } return qfalse; } extern void CG_LandingEffect( vec3_t origin, vec3_t normal, int material ); static void PM_CrashLandEffect( void ) { if ( pm->waterlevel ) { return; } float delta = fabs(pml.previous_velocity[2])/10;//VectorLength( pml.previous_velocity );? if ( delta >= 30 ) { vec3_t bottom = {pm->ps->origin[0],pm->ps->origin[1],pm->ps->origin[2]+pm->mins[2]+1}; CG_LandingEffect( bottom, pml.groundTrace.plane.normal, (pml.groundTrace.surfaceFlags&MATERIAL_MASK) ); } } /* ================= PM_CrashLand Check for hard landings that generate sound events ================= */ static void PM_CrashLand( void ) { float delta = 0; qboolean forceLanding = qfalse; if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_VEHICLE ) { if ( pm->gent->m_pVehicle->m_pVehicleInfo->type != VH_ANIMAL ) { float dot = DotProduct( pm->ps->velocity, pml.groundTrace.plane.normal ); //Com_Printf("%i:crashland %4.2f\n", c_pmove, pm->ps->velocity[2]); if ( dot < -100.0f ) { //NOTE: never hits this anyway if ( pm->gent->m_pVehicle->m_pVehicleInfo->iImpactFX ) {//make sparks if ( !Q_irand( 0, 3 ) ) {//FIXME: debounce G_PlayEffect( pm->gent->m_pVehicle->m_pVehicleInfo->iImpactFX, pm->ps->origin, pml.groundTrace.plane.normal ); } } int damage = floor(fabs(dot+100)/10.0f); if ( damage >= 0 ) { G_Damage( pm->gent, NULL, NULL, NULL, NULL, damage, 0, MOD_FALLING ); } } } return; } if ( (pm->ps->pm_flags&PMF_TRIGGER_PUSHED) ) { delta = 21;//? forceLanding = qtrue; } else { if ( pm->gent && pm->gent->NPC && pm->gent->NPC->aiFlags & NPCAI_DIE_ON_IMPACT ) {//have to do death on impact if we are falling to our death, FIXME: should we avoid any additional damage this func? PM_CrashLandDamage( 1000 ); } else if ( pm->gent && pm->gent->client && (pm->gent->client->NPC_class == CLASS_BOBAFETT||pm->gent->client->NPC_class == CLASS_ROCKETTROOPER) ) { if ( JET_Flying( pm->gent ) ) { if ( pm->gent->client->NPC_class == CLASS_BOBAFETT || (pm->gent->client->NPC_class == CLASS_ROCKETTROOPER&&pm->gent->NPC&&pm->gent->NPC->rankgent ); } else { pm->ps->velocity[2] += Q_flrand( 100, 200 ); } PM_AddEvent( EV_FALL_SHORT ); } if ( pm->ps->forceJumpZStart ) {//we were force-jumping forceLanding = qtrue; } delta = 1; } else if ( pm->ps->jumpZStart && (pm->ps->forcePowerLevel[FP_LEVITATION] >= FORCE_LEVEL_1||(pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer())) ) {//we were force-jumping if ( pm->ps->origin[2] >= pm->ps->jumpZStart ) {//we landed at same height or higher than we landed if ( pm->ps->forceJumpZStart ) {//we were force-jumping forceLanding = qtrue; } delta = 0; } else {//take off some of it, at least delta = (pm->ps->jumpZStart-pm->ps->origin[2]); float dropAllow = forceJumpHeight[pm->ps->forcePowerLevel[FP_LEVITATION]]; if ( dropAllow < 128 ) {//always allow a drop from 128, at least dropAllow = 128; } if ( delta > forceJumpHeight[FORCE_LEVEL_1] ) {//will have to use force jump ability to absorb some of it forceLanding = qtrue;//absorbed some - just to force the correct animation to play below } delta = (delta - dropAllow)/2; } if ( delta < 1 ) { delta = 1; } } if ( !delta ) { delta = PM_CrashLandDelta( pml.previous_velocity, pm->waterlevel ); } } PM_CrashLandEffect(); if ( (pm->ps->pm_flags&PMF_DUCKED) && (level.time-pm->ps->lastOnGround)>500 ) {//must be crouched and have been inthe air for half a second minimum if( !PM_InOnGroundAnim( pm->ps ) && !PM_InKnockDown( pm->ps ) ) {//roll! if ( PM_TryRoll() ) {//absorb some impact delta *= 0.5f; } } } // If he just entered the water (from a fall presumably), absorb some of the impact. if ( pm->waterlevel >= 2 ) { delta *= 0.4f; } if ( delta < 1 ) { AddSoundEvent( pm->gent, pm->ps->origin, 32, AEL_MINOR, qfalse, qtrue ); return; } qboolean deadFallSound = qfalse; if( !PM_InDeathAnim() ) { if ( PM_InAirKickingAnim( pm->ps->legsAnim ) && pm->ps->torsoAnim == pm->ps->legsAnim ) { int anim = PM_GetLandingAnim(); if ( anim != -1 ) {//interrupting a kick clears everything PM_SetAnim( pm, SETANIM_BOTH, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD, 100 ); // Only blend over 100ms pm->ps->saberMove = LS_READY; pm->ps->weaponTime = 0; if ( !g_allowBunnyhopping->integer ) { //stick landings some pm->ps->velocity[0] *= 0.5f; pm->ps->velocity[1] *= 0.5f; } } } else if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_ROCKETTROOPER ) {//rockettroopers are simpler int anim = PM_GetLandingAnim(); if ( anim != -1 ) { if ( pm->gent->NPC && pm->gent->NPC->rank < RANK_LT ) {//special case: ground-based rocket troopers *always* play land anim on whole body PM_SetAnim( pm, SETANIM_BOTH, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD, 100 ); // Only blend over 100ms } else { PM_SetAnim( pm, SETANIM_LEGS, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD, 100 ); // Only blend over 100ms } } } else if ( pm->cmd.upmove >= 0 && !PM_InKnockDown( pm->ps ) && !PM_InRoll( pm->ps )) {//not crouching if ( delta > 10 || pm->ps->pm_flags & PMF_BACKWARDS_JUMP || (pm->ps->forcePowersActive&(1<gent && pm->gent->client && (pm->gent->client->NPC_class == CLASS_RANCOR || pm->gent->client->NPC_class == CLASS_WAMPA ) ) { } else { int anim = PM_GetLandingAnim(); if ( anim != -1 ) { if ( PM_FlippingAnim( pm->ps->torsoAnim ) || PM_SpinningAnim( pm->ps->torsoAnim ) || pm->ps->torsoAnim == BOTH_FLIP_LAND ) {//interrupt these if we're going to play a land pm->ps->torsoAnimTimer = 0; } if ( anim == BOTH_FORCELONGLEAP_LAND ) { if ( pm->gent ) { G_SoundOnEnt( pm->gent, CHAN_AUTO, "sound/player/slide.wav" ); } PM_SetAnim( pm, SETANIM_BOTH, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD, 100 ); // Only blend over 100ms } else if ( anim == BOTH_FLIP_LAND || (pm->ps->torsoAnim == BOTH_FLIP_LAND) ) { PM_SetAnim( pm, SETANIM_BOTH, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD, 100 ); // Only blend over 100ms } else if ( PM_InAirKickingAnim( pm->ps->legsAnim ) && pm->ps->torsoAnim == pm->ps->legsAnim ) {//interrupting a kick clears everything PM_SetAnim( pm, SETANIM_BOTH, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD, 100 ); // Only blend over 100ms pm->ps->saberMove = LS_READY; pm->ps->weaponTime = 0; } else if ( PM_ForceJumpingAnim( pm->ps->legsAnim ) && pm->ps->torsoAnim == pm->ps->legsAnim ) {//special case: if land during one of these, set the torso, too, if it's not doing something else PM_SetAnim( pm, SETANIM_BOTH, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD, 100 ); // Only blend over 100ms } else { PM_SetAnim( pm, SETANIM_LEGS, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD, 100 ); // Only blend over 100ms } } } } } } else { pm->ps->gravity = 1.0; //PM_CrashLandDamage( delta ); if ( pm->gent ) { if ((!(pml.groundTrace.surfaceFlags & SURF_NODAMAGE)) && // (!(pml.groundTrace.contents & CONTENTS_NODROP)) && (!(pm->pointcontents(pm->ps->origin,pm->ps->clientNum) & CONTENTS_NODROP))) { if ( pm->waterlevel < 2 ) {//don't play fallsplat when impact in the water deadFallSound = qtrue; if ( !(pm->ps->eFlags&EF_NODRAW) ) {//no sound if no draw if ( delta >= 75 ) { G_SoundOnEnt( pm->gent, CHAN_BODY, "sound/player/fallsplat.wav" ); } else { G_SoundOnEnt( pm->gent, CHAN_BODY, va("sound/player/bodyfall_human%d.wav",Q_irand(1,3)) ); } } } else { G_SoundOnEnt( pm->gent, CHAN_BODY, va("sound/player/bodyfall_water%d.wav",Q_irand(1,3)) ); } if ( gi.VoiceVolume[pm->ps->clientNum] && pm->gent->NPC && (pm->gent->NPC->aiFlags&NPCAI_DIE_ON_IMPACT) ) {//I was talking, so cut it off... with a jump sound? if ( !(pm->ps->eFlags&EF_NODRAW) ) {//no sound if no draw G_SoundOnEnt( pm->gent, CHAN_VOICE_ATTEN, "*pain100.wav" ); } } } } if( pm->ps->legsAnim == BOTH_FALLDEATH1 || pm->ps->legsAnim == BOTH_FALLDEATH1INAIR) {//FIXME: add a little bounce? //FIXME: cut voice channel? int old_pm_type = pm->ps->pm_type; pm->ps->pm_type = PM_NORMAL; //Hack because for some reason PM_SetAnim just returns if you're dead...??? PM_SetAnim(pm, SETANIM_BOTH, BOTH_FALLDEATH1LAND, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD); pm->ps->pm_type = old_pm_type; AddSoundEvent( pm->gent, pm->ps->origin, 256, AEL_SUSPICIOUS, qfalse, qtrue ); return; } } // create a local entity event to play the sound if ( pm->gent && pm->gent->client && pm->gent->client->respawnTime >= level.time - 500 ) {//just spawned in, don't make a noise return; } if ( delta >= 75 ) { if ( !deadFallSound ) { if ( !(pm->ps->eFlags&EF_NODRAW) ) {//no sound if no draw PM_AddEvent( EV_FALL_FAR ); } } if ( !(pml.groundTrace.surfaceFlags & SURF_NODAMAGE) ) { PM_CrashLandDamage( delta ); } if ( pm->gent ) { if ( pm->gent->s.number == 0 ) { vec3_t bottom; VectorCopy( pm->ps->origin, bottom ); bottom[2] += pm->mins[2]; // if ( pm->gent->client && pm->gent->client->playerTeam != TEAM_DISGUISE ) { AddSoundEvent( pm->gent, bottom, 256, AEL_SUSPICIOUS, qfalse, qtrue ); } } else if ( pm->ps->stats[STAT_HEALTH] <= 0 && pm->gent && pm->gent->enemy ) { AddSoundEvent( pm->gent->enemy, pm->ps->origin, 256, AEL_DISCOVERED, qfalse, qtrue ); } } } else if ( delta >= 50 ) { // this is a pain grunt, so don't play it if dead if ( pm->ps->stats[STAT_HEALTH] > 0 ) { if ( !deadFallSound ) { if ( !(pm->ps->eFlags&EF_NODRAW) ) {//no sound if no draw PM_AddEvent( EV_FALL_MEDIUM );//damage is dealt in g_active, ClientEvents } } if ( pm->gent ) { if ( !(pml.groundTrace.surfaceFlags & SURF_NODAMAGE) ) { PM_CrashLandDamage( delta ); } if ( pm->gent->s.number == 0 ) { vec3_t bottom; VectorCopy( pm->ps->origin, bottom ); bottom[2] += pm->mins[2]; // if ( pm->gent->client && pm->gent->client->playerTeam != TEAM_DISGUISE ) { AddSoundEvent( pm->gent, bottom, 256, AEL_MINOR, qfalse, qtrue ); } } } } } else if ( delta >= 30 ) { if ( !deadFallSound ) { if ( !(pm->ps->eFlags&EF_NODRAW) ) {//no sound if no draw PM_AddEvent( EV_FALL_SHORT ); } } if ( pm->gent ) { if ( pm->gent->s.number == 0 ) { vec3_t bottom; VectorCopy( pm->ps->origin, bottom ); bottom[2] += pm->mins[2]; // if ( pm->gent->client && pm->gent->client->playerTeam != TEAM_DISGUISE ) { AddSoundEvent( pm->gent, bottom, 128, AEL_MINOR, qfalse, qtrue ); } } else { if ( !(pml.groundTrace.surfaceFlags & SURF_NODAMAGE) ) { PM_CrashLandDamage( delta ); } } } } else { if ( !deadFallSound ) { if ( !(pm->ps->pm_flags&PMF_DUCKED) || !Q_irand( 0, 3 ) ) {//only 25% chance of making landing alert when ducked AddSoundEvent( pm->gent, pm->ps->origin, 32, AEL_MINOR, qfalse, qtrue ); } if ( !(pm->ps->eFlags&EF_NODRAW) ) {//no sound if no draw if ( forceLanding ) {//we were force-jumping PM_AddEvent( EV_FALL_SHORT ); } else { //moved to cg_player PM_AddEvent( PM_FootstepForSurface() ); } } } } // start footstep cycle over pm->ps->bobCycle = 0; if ( pm->gent && pm->gent->client ) {//stop the force push effect when you land pm->gent->forcePushTime = 0; } } /* ============= PM_CorrectAllSolid ============= */ static void PM_CorrectAllSolid( void ) { if ( pm->debugLevel ) { Com_Printf("%i:allsolid\n", c_pmove); //NOTENOTE: If this ever happens, I'd really like to see this print! } // FIXME: jitter around pm->ps->groundEntityNum = ENTITYNUM_NONE; pml.groundPlane = qfalse; pml.walking = qfalse; } qboolean FlyingCreature( gentity_t *ent ) { if ( ent->client->ps.gravity <= 0 && (ent->svFlags&SVF_CUSTOM_GRAVITY) ) { return qtrue; } return qfalse; } qboolean PM_RocketeersAvoidDangerousFalls( void ) { if ( pm->gent->NPC && pm->gent->client && (pm->gent->client->NPC_class == CLASS_BOBAFETT||pm->gent->client->NPC_class == CLASS_ROCKETTROOPER) ) {//fixme: fall through if jetpack broken? if ( JET_Flying( pm->gent ) ) { if ( pm->gent->client->NPC_class == CLASS_BOBAFETT ) { pm->gent->client->jetPackTime = level.time + 2000; //Wait, what if the effect is already playing, how do we know??? //G_PlayEffect( G_EffectIndex( "boba/jetSP" ), pm->gent->playerModel, pm->gent->genericBolt1, pm->gent->s.number, pm->gent->currentOrigin, pm->gent->client->jetPackTime-level.time ); } else { pm->gent->client->jetPackTime = Q3_INFINITE; } } else { TIMER_Set( pm->gent, "jetRecharge", 0 ); JET_FlyStart( pm->gent ); } return qtrue; } return qfalse; } static void PM_FallToDeath( void ) { if ( !pm->gent ) { return; } if ( PM_RocketeersAvoidDangerousFalls() ) { return; } // If this is a vehicle, more precisely an animal... if ( pm->gent->client->NPC_class == CLASS_VEHICLE && pm->gent->m_pVehicle->m_pVehicleInfo->type == VH_ANIMAL ) { Vehicle_t *pVeh = pm->gent->m_pVehicle; pVeh->m_pVehicleInfo->EjectAll( pVeh ); } else { if ( PM_HasAnimation( pm->gent, BOTH_FALLDEATH1 ) ) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_FALLDEATH1,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD); } else { PM_SetAnim(pm,SETANIM_LEGS,BOTH_DEATH1,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD); } G_SoundOnEnt( pm->gent, CHAN_VOICE, "*falling1.wav" );//CHAN_VOICE_ATTEN } if ( pm->gent->NPC ) { pm->gent->NPC->aiFlags |= NPCAI_DIE_ON_IMPACT; pm->gent->NPC->nextBStateThink = Q3_INFINITE; } pm->ps->friction = 1; } int PM_ForceJumpAnimForJumpAnim( int anim ) { switch( anim ) { case BOTH_JUMP1: //# Jump - wind-up and leave ground anim = BOTH_FORCEJUMP1; //# Jump - wind-up and leave ground break; case BOTH_INAIR1: //# In air loop (from jump) anim = BOTH_FORCEINAIR1; //# In air loop (from jump) break; case BOTH_LAND1: //# Landing (from in air loop) anim = BOTH_FORCELAND1; //# Landing (from in air loop) break; case BOTH_JUMPBACK1: //# Jump backwards - wind-up and leave ground anim = BOTH_FORCEJUMPBACK1; //# Jump backwards - wind-up and leave ground break; case BOTH_INAIRBACK1: //# In air loop (from jump back) anim = BOTH_FORCEINAIRBACK1; //# In air loop (from jump back) break; case BOTH_LANDBACK1: //# Landing backwards(from in air loop) anim = BOTH_FORCELANDBACK1; //# Landing backwards(from in air loop) break; case BOTH_JUMPLEFT1: //# Jump left - wind-up and leave ground anim = BOTH_FORCEJUMPLEFT1; //# Jump left - wind-up and leave ground break; case BOTH_INAIRLEFT1: //# In air loop (from jump left) anim = BOTH_FORCEINAIRLEFT1; //# In air loop (from jump left) break; case BOTH_LANDLEFT1: //# Landing left(from in air loop) anim = BOTH_FORCELANDLEFT1; //# Landing left(from in air loop) break; case BOTH_JUMPRIGHT1: //# Jump right - wind-up and leave ground anim = BOTH_FORCEJUMPRIGHT1; //# Jump right - wind-up and leave ground break; case BOTH_INAIRRIGHT1: //# In air loop (from jump right) anim = BOTH_FORCEINAIRRIGHT1; //# In air loop (from jump right) break; case BOTH_LANDRIGHT1: //# Landing right(from in air loop) anim = BOTH_FORCELANDRIGHT1; //# Landing right(from in air loop) break; } return anim; } static void PM_SetVehicleAngles( vec3_t normal ) { if ( !pm->gent->client || pm->gent->client->NPC_class != CLASS_VEHICLE ) return; Vehicle_t *pVeh = pm->gent->m_pVehicle; //float curVehicleBankingSpeed; float vehicleBankingSpeed = pVeh->m_pVehicleInfo->bankingSpeed;//0.25f vec3_t vAngles; if ( vehicleBankingSpeed <= 0 || ( pVeh->m_pVehicleInfo->pitchLimit <= 0 && pVeh->m_pVehicleInfo->rollLimit <= 0 ) ) {//don't bother, this vehicle doesn't bank return; } //FIXME: do 3 traces to define a plane and use that... smoothes it out some, too... //pitch_roll_for_slope( pm->gent, normal, vAngles ); //FIXME: maybe have some pitch control in water and/or air? //center of gravity affects pitch in air/water (FIXME: what about roll?) //float pitchBias = 90.0f*pVeh->m_pVehicleInfo->centerOfGravity[0];//if centerOfGravity is all the way back (-1.0f), vehicle pitches up 90 degrees when in air VectorClear( vAngles ); if (pm->waterlevel>0 || (normal && (pml.groundTrace.contents&(CONTENTS_WATER|CONTENTS_SLIME|CONTENTS_LAVA)))) {//in water // vAngles[PITCH] += (pm->ps->viewangles[PITCH]-vAngles[PITCH])*0.75f + (pitchBias*0.5); } else if ( normal ) {//have a valid surface below me pitch_roll_for_slope( pm->gent, normal, vAngles ); float deltaPitch = (vAngles[PITCH] - pVeh->m_vOrientation[PITCH]); if (deltaPitch< -10.0f) { vAngles[PITCH] = pVeh->m_vOrientation[PITCH] - 10.0f; } else if (deltaPitch>10.0f) { vAngles[PITCH] = pVeh->m_vOrientation[PITCH] + 10.0f; } } else {//in air, let pitch match view...? //FIXME: take center of gravity into account vAngles[PITCH] = pVeh->m_vOrientation[PITCH] - 1; if (vAngles[PITCH]<-15) { vAngles[PITCH]=-15; } //don't bank so fast when in the air vehicleBankingSpeed *= 0.125f; } //NOTE: if angles are flat and we're moving through air (not on ground), // then pitch/bank? if (pVeh->m_ulFlags & VEH_SPINNING) { vAngles[ROLL] = pVeh->m_vOrientation[ROLL] - 25; } else if (pVeh->m_ulFlags & VEH_OUTOFCONTROL) { //vAngles[ROLL] = pVeh->m_vOrientation[ROLL] + 5; } else if ( pVeh->m_pVehicleInfo->rollLimit > 0 ) { //roll when banking vec3_t velocity; float speed; VectorCopy( pm->ps->velocity, velocity ); speed = VectorNormalize( velocity ); if ( speed>0.01f ) { vec3_t rt, tempVAngles; float side, dotp; VectorCopy( pVeh->m_vOrientation, tempVAngles ); tempVAngles[ROLL] = 0; AngleVectors( tempVAngles, NULL, rt, NULL ); dotp = DotProduct( velocity, rt ); //if (fabsf(dotp)>0.5f) { speed *= dotp; // Magic number fun! Speed is used for banking, so modulate the speed by a sine wave //FIXME: this banks too early //speed *= sin( (150 + pml.frametime) * 0.003 ); if (level.time < pVeh->m_iTurboTime) { speed /= pVeh->m_pVehicleInfo->turboSpeed; } else { speed /= pVeh->m_pVehicleInfo->speedMax; } /// if (pm->cmd.forwardmove==0) // { // speed *= 0.5; // } // if (pm->cmd.forwardmove<0) // { // speed *= 0.1f; // } if (pVeh->m_ulFlags & VEH_SLIDEBREAKING) { speed *= 3.0f; } side = speed * 75.0f; // if (pVeh->m_ulFlags & VEH_STRAFERAM) // { // vAngles[ROLL] += side; // } // else { vAngles[ROLL] -= side; } //gi.Printf("VAngles(%f)", vAngles[2]); } if (fabsf(vAngles[ROLL])<0.001f) { vAngles[ROLL] = 0.0f; } } } //cap if ( vAngles[PITCH] > pVeh->m_pVehicleInfo->pitchLimit ) { vAngles[PITCH] = pVeh->m_pVehicleInfo->pitchLimit; } else if ( vAngles[PITCH] < -pVeh->m_pVehicleInfo->pitchLimit ) { vAngles[PITCH] = -pVeh->m_pVehicleInfo->pitchLimit; } if (!(pVeh->m_ulFlags & VEH_SPINNING)) { if ( vAngles[ROLL] > pVeh->m_pVehicleInfo->rollLimit ) { vAngles[ROLL] = pVeh->m_pVehicleInfo->rollLimit; } else if ( vAngles[ROLL] < -pVeh->m_pVehicleInfo->rollLimit ) { vAngles[ROLL] = -pVeh->m_pVehicleInfo->rollLimit; } } //do it for ( int i = 0; i < 3; i++ ) { if ( i == YAW ) {//yawing done elsewhere continue; } if ( i == ROLL && pVeh->m_ulFlags & VEH_STRAFERAM) {//during strafe ram, roll is done elsewhere continue; } //bank faster the higher the difference is /* else if ( i == PITCH ) { curVehicleBankingSpeed = vehicleBankingSpeed*fabs(AngleNormalize180(AngleSubtract( vAngles[PITCH], pVeh->m_vOrientation[PITCH] )))/(g_vehicleInfo[pm->ps->vehicleIndex].pitchLimit/2.0f); } else if ( i == ROLL ) { curVehicleBankingSpeed = vehicleBankingSpeed*fabs(AngleNormalize180(AngleSubtract( vAngles[ROLL], pVeh->m_vOrientation[ROLL] )))/(g_vehicleInfo[pm->ps->vehicleIndex].rollLimit/2.0f); } if ( curVehicleBankingSpeed ) */ { // if ( pVeh->m_vOrientation[i] >= vAngles[i] + vehicleBankingSpeed ) // { // pVeh->m_vOrientation[i] -= vehicleBankingSpeed; // } // else if ( pVeh->m_vOrientation[i] <= vAngles[i] - vehicleBankingSpeed ) // { // pVeh->m_vOrientation[i] += vehicleBankingSpeed; // } // else { pVeh->m_vOrientation[i] = vAngles[i]; } } } //gi.Printf("Orientation(%f)", pVeh->m_vOrientation[2]); } void BG_ExternThisSoICanRecompileInDebug( Vehicle_t *pVeh, playerState_t *riderPS ) {/* float pitchSubtract, pitchDelta, yawDelta; //Com_Printf( S_COLOR_RED"PITCH: %4.2f, YAW: %4.2f, ROLL: %4.2f\n", riderPS->viewangles[0],riderPS->viewangles[1],riderPS->viewangles[2]); yawDelta = AngleSubtract(riderPS->viewangles[YAW],pVeh->m_vPrevRiderViewAngles[YAW]); #ifndef QAGAME if ( !cg_paused.integer ) { //Com_Printf( "%d - yawDelta %4.2f\n", pm->cmd.serverTime, yawDelta ); } #endif yawDelta *= (4.0f*pVeh->m_fTimeModifier); pVeh->m_vOrientation[ROLL] -= yawDelta; pitchDelta = AngleSubtract(riderPS->viewangles[PITCH],pVeh->m_vPrevRiderViewAngles[PITCH]); pitchDelta *= (2.0f*pVeh->m_fTimeModifier); pitchSubtract = pitchDelta * (fabs(pVeh->m_vOrientation[ROLL])/90.0f); pVeh->m_vOrientation[PITCH] += pitchDelta-pitchSubtract; if ( pVeh->m_vOrientation[ROLL] > 0 ) { pVeh->m_vOrientation[YAW] += pitchSubtract; } else { pVeh->m_vOrientation[YAW] -= pitchSubtract; } pVeh->m_vOrientation[PITCH] = AngleNormalize180( pVeh->m_vOrientation[PITCH] ); pVeh->m_vOrientation[YAW] = AngleNormalize360( pVeh->m_vOrientation[YAW] ); pVeh->m_vOrientation[ROLL] = AngleNormalize180( pVeh->m_vOrientation[ROLL] ); VectorCopy( riderPS->viewangles, pVeh->m_vPrevRiderViewAngles );*/ } void BG_VehicleTurnRateForSpeed( Vehicle_t *pVeh, float speed, float *mPitchOverride, float *mYawOverride ) { if ( pVeh && pVeh->m_pVehicleInfo ) { float speedFrac = 1.0f; if ( pVeh->m_pVehicleInfo->speedDependantTurning ) { if ( pVeh->m_LandTrace.fraction >= 1.0f || pVeh->m_LandTrace.plane.normal[2] < MIN_LANDING_SLOPE ) { speedFrac = (speed/(pVeh->m_pVehicleInfo->speedMax*0.75f)); if ( speedFrac < 0.25f ) { speedFrac = 0.25f; } else if ( speedFrac > 1.0f ) { speedFrac = 1.0f; } } } if ( pVeh->m_pVehicleInfo->mousePitch ) { *mPitchOverride = pVeh->m_pVehicleInfo->mousePitch*speedFrac; } if ( pVeh->m_pVehicleInfo->mouseYaw ) { *mYawOverride = pVeh->m_pVehicleInfo->mouseYaw*speedFrac; } } } /* ============= PM_GroundTraceMissed The ground trace didn't hit a surface, so we are in freefall ============= */ static void PM_GroundTraceMissed( void ) { trace_t trace; vec3_t point; qboolean cliff_fall = qfalse; if ( Flying != FLY_HOVER ) { if ( !(pm->ps->eFlags&EF_FORCE_DRAINED) ) { //FIXME: if in a contents_falldeath brush, play the falling death anim and sound? if ( pm->ps->clientNum != 0 && pm->gent && pm->gent->NPC && pm->gent->client && pm->gent->client->NPC_class != CLASS_DESANN )//desann never falls to his death { if ( pm->ps->groundEntityNum == ENTITYNUM_NONE ) { if ( pm->ps->stats[STAT_HEALTH] > 0 && !(pm->gent->NPC->aiFlags&NPCAI_DIE_ON_IMPACT) && !(pm->gent->NPC->aiFlags&NPCAI_JUMP) // doing a path jump && !(pm->gent->NPC->scriptFlags&SCF_NO_FALLTODEATH) && pm->gent->NPC->behaviorState != BS_JUMP )//not being scripted to jump { if ( (level.time - pm->gent->client->respawnTime > 2000)//been in the world for at least 2 seconds && (!pm->gent->NPC->timeOfDeath || level.time - pm->gent->NPC->timeOfDeath < 1000) && pm->gent->e_ThinkFunc != thinkF_NPC_RemoveBody //Have to do this now because timeOfDeath is used by thinkF_NPC_RemoveBody to debounce removal checks && !(pm->gent->client->ps.forcePowersActive&(1<gent ) && g_gravity->value > 0 ) { if ( !(pm->gent->flags&FL_UNDYING) && !(pm->gent->flags&FL_GODMODE) ) { if ( !(pm->ps->eFlags&EF_FORCE_GRIPPED) && !(pm->ps->eFlags&EF_FORCE_DRAINED) && !(pm->ps->pm_flags&PMF_TRIGGER_PUSHED) ) { if ( !pm->ps->forceJumpZStart || pm->ps->forceJumpZStart > pm->ps->origin[2] )// && fabs(pm->ps->velocity[0])<10 && fabs(pm->ps->velocity[1])<10 && pm->ps->velocity[2]<0)//either not force-jumping or force-jumped and now fell below original jump start height { /*if ( pm->ps->legsAnim = BOTH_FALLDEATH1 && pm->ps->legsAnim != BOTH_DEATH1 && PM_HasAnimation( pm->gent, BOTH_FALLDEATH1 )*/ //New method: predict impact, 400 ahead vec3_t vel; float time; VectorCopy( pm->ps->velocity, vel ); float speed = VectorLength( vel ); if ( !speed ) {//damn divide by zero speed = 1; } time = 400/speed; vel[2] -= 0.5 * time * pm->ps->gravity; speed = VectorLength( vel ); if ( !speed ) {//damn divide by zero speed = 1; } time = 400/speed; VectorScale( vel, time, vel ); VectorAdd( pm->ps->origin, vel, point ); pm->trace( &trace, pm->ps->origin, pm->mins, pm->maxs, point, pm->ps->clientNum, pm->tracemask, (EG2_Collision)0, 0 ); if ( (trace.contents&CONTENTS_LAVA) && PM_RocketeersAvoidDangerousFalls() ) {//got out of it } else if ( !trace.allsolid && !trace.startsolid && (pm->ps->origin[2] - trace.endpos[2]) >= 128 )//>=128 so we don't die on steps! { if ( trace.fraction == 1.0 ) {//didn't hit, we're probably going to die if ( pm->ps->velocity[2] < 0 && pm->ps->origin[2] - point[2] > 256 ) {//going down, into a bottomless pit, apparently PM_FallToDeath(); cliff_fall = qtrue; } } else if ( trace.entityNum < ENTITYNUM_NONE && pm->ps->weapon != WP_SABER && (!pm->gent || !pm->gent->client || (pm->gent->client->NPC_class != CLASS_BOBAFETT&&pm->gent->client->NPC_class!=CLASS_REBORN&&pm->gent->client->NPC_class!=CLASS_ROCKETTROOPER)) ) {//Jedi don't scream and die if they're heading for a hard impact gentity_t *traceEnt = &g_entities[trace.entityNum]; if ( trace.entityNum == ENTITYNUM_WORLD || (traceEnt && traceEnt->bmodel) ) {//hit architecture, find impact force float dmg; VectorCopy( pm->ps->velocity, vel ); time = Distance( trace.endpos, pm->ps->origin )/VectorLength( vel ); vel[2] -= 0.5 * time * pm->ps->gravity; if ( trace.plane.normal[2] > 0.5 ) {//use falling damage int waterlevel, junk; PM_SetWaterLevelAtPoint( trace.endpos, &waterlevel, &junk ); dmg = PM_CrashLandDelta( vel, waterlevel ); if ( dmg >= 30 ) {//there is a minimum fall threshhold dmg = PM_DamageForDelta( dmg ); } else { dmg = 0; } } else {//use impact damage //guestimate if ( pm->gent->client && pm->gent->client->ps.forceJumpZStart ) {//we were force-jumping if ( pm->gent->currentOrigin[2] >= pm->gent->client->ps.forceJumpZStart ) {//we landed at same height or higher than we landed dmg = 0; } else {//FIXME: take off some of it, at least? dmg = (pm->gent->client->ps.forceJumpZStart-pm->gent->currentOrigin[2])/3; } } dmg = 10 * VectorLength( pm->ps->velocity ); if ( pm->ps->clientNum < MAX_CLIENTS ) { dmg /= 2; } dmg *= 0.01875f;//magic number } float maxDmg = pm->ps->stats[STAT_HEALTH]>20?pm->ps->stats[STAT_HEALTH]:20;//a fall that would do less than 20 points of damage should never make us scream to our deaths if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_HOWLER ) {//howlers can survive long falls, despire their health maxDmg *= 5; } if ( dmg >= pm->ps->stats[STAT_HEALTH] )//armor? { PM_FallToDeath(); cliff_fall = qtrue; } } } } /* vec3_t start; //okay, kind of expensive temp hack here, but let's check to see if we should scream //FIXME: we should either do a better check (predict using actual velocity) or we should wait until they've been over a bottomless pit for a certain amount of time... VectorCopy( pm->ps->origin, start ); if ( pm->ps->forceJumpZStart < start[2] ) {//Jedi who are force-jumping should only do this from landing point down? start[2] = pm->ps->forceJumpZStart; } VectorCopy( start, point ); point[2] -= 400;//320 pm->trace (&trace, pm->ps->origin, pm->mins, pm->maxs, point, pm->ps->clientNum, pm->tracemask); //FIXME: somehow, people can still get stuck on ledges and not splat when hit...? if ( !trace.allsolid && !trace.startsolid && trace.fraction == 1.0 ) { PM_FallToDeath(); cliff_fall = qtrue; } */ } } } } } } } } if ( !cliff_fall ) { if ( pm->ps->legsAnim == BOTH_FORCELONGLEAP_START || pm->ps->legsAnim == BOTH_FORCELONGLEAP_ATTACK || pm->ps->legsAnim == BOTH_FORCEWALLRUNFLIP_START || pm->ps->legsAnim == BOTH_FLIP_LAND ) {//let it stay on this anim } else if ( PM_KnockDownAnimExtended( pm->ps->legsAnim ) ) {//no in-air anims when in knockdown anim } else if ( ( pm->ps->legsAnim == BOTH_WALL_RUN_RIGHT || pm->ps->legsAnim == BOTH_WALL_RUN_LEFT || pm->ps->legsAnim == BOTH_WALL_RUN_RIGHT_STOP || pm->ps->legsAnim == BOTH_WALL_RUN_LEFT_STOP || pm->ps->legsAnim == BOTH_WALL_RUN_RIGHT_FLIP || pm->ps->legsAnim == BOTH_WALL_RUN_LEFT_FLIP || pm->ps->legsAnim == BOTH_WALL_FLIP_RIGHT || pm->ps->legsAnim == BOTH_WALL_FLIP_LEFT || pm->ps->legsAnim == BOTH_FORCEWALLREBOUND_FORWARD || pm->ps->legsAnim == BOTH_FORCEWALLREBOUND_LEFT || pm->ps->legsAnim == BOTH_FORCEWALLREBOUND_BACK || pm->ps->legsAnim == BOTH_FORCEWALLREBOUND_RIGHT || pm->ps->legsAnim == BOTH_CEILING_DROP ) && !pm->ps->legsAnimTimer ) {//if flip anim is done, okay to use inair PM_SetAnim( pm, SETANIM_LEGS, BOTH_FORCEINAIR1, SETANIM_FLAG_OVERRIDE, 350 ); // Only blend over 100ms } else if ( pm->ps->legsAnim == BOTH_FLIP_ATTACK7 || pm->ps->legsAnim == BOTH_FLIP_HOLD7 ) { if ( !pm->ps->legsAnimTimer ) {//done? PM_SetAnim( pm, SETANIM_BOTH, BOTH_FLIP_HOLD7, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD, 350 ); // Only blend over 100ms } } else if ( PM_InCartwheel( pm->ps->legsAnim ) ) { if ( pm->ps->legsAnimTimer > 0 ) {//still playing on bottom } else { PM_SetAnim( pm, SETANIM_BOTH, BOTH_INAIR1, SETANIM_FLAG_NORMAL, 350 ); } } else if ( PM_InAirKickingAnim( pm->ps->legsAnim ) ) { if ( pm->ps->legsAnimTimer > 0 ) {//still playing on bottom } else { PM_SetAnim( pm, SETANIM_BOTH, BOTH_INAIR1, SETANIM_FLAG_NORMAL, 350 ); pm->ps->saberMove = LS_READY; pm->ps->weaponTime = 0; } } else if ( !PM_InRoll( pm->ps ) && !PM_SpinningAnim( pm->ps->legsAnim ) && !PM_FlippingAnim( pm->ps->legsAnim ) && !PM_InSpecialJump( pm->ps->legsAnim ) ) { if ( pm->ps->groundEntityNum != ENTITYNUM_NONE ) { // we just transitioned into freefall if ( pm->debugLevel ) { Com_Printf("%i:lift\n", c_pmove); } // if they aren't in a jumping animation and the ground is a ways away, force into it // if we didn't do the trace, the player would be backflipping down staircases VectorCopy( pm->ps->origin, point ); point[2] -= 64; pm->trace (&trace, pm->ps->origin, pm->mins, pm->maxs, point, pm->ps->clientNum, pm->tracemask, (EG2_Collision)0, 0); if ( trace.fraction == 1.0 ) {//FIXME: if velocity[2] < 0 and didn't jump, use some falling anim if ( pm->ps->velocity[2] <= 0 && !(pm->ps->pm_flags&PMF_JUMP_HELD)) { if(!PM_InDeathAnim()) { // If we're a vehicle, set our falling flag. if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_VEHICLE ) { // We're flying in the air. if ( pm->gent->m_pVehicle->m_pVehicleInfo->type == VH_ANIMAL ) { pm->gent->m_pVehicle->m_ulFlags |= VEH_FLYING; } } else { vec3_t moveDir, lookAngles, lookDir, lookRight; int anim = BOTH_INAIR1; VectorCopy( pm->ps->velocity, moveDir ); moveDir[2] = 0; VectorNormalize( moveDir ); VectorCopy( pm->ps->viewangles, lookAngles ); lookAngles[PITCH] = lookAngles[ROLL] = 0; AngleVectors( lookAngles, lookDir, lookRight, NULL ); float dot = DotProduct( moveDir, lookDir ); if ( dot > 0.5 ) {//redundant anim = BOTH_INAIR1; } else if ( dot < -0.5 ) { anim = BOTH_INAIRBACK1; } else { dot = DotProduct( moveDir, lookRight ); if ( dot > 0.5 ) { anim = BOTH_INAIRRIGHT1; } else if ( dot < -0.5 ) { anim = BOTH_INAIRLEFT1; } else {//redundant anim = BOTH_INAIR1; } } if ( pm->ps->forcePowersActive & ( 1 << FP_LEVITATION ) ) { anim = PM_ForceJumpAnimForJumpAnim( anim ); } PM_SetAnim( pm, SETANIM_LEGS, anim, SETANIM_FLAG_OVERRIDE, 100 ); // Only blend over 100ms } } pm->ps->pm_flags &= ~PMF_BACKWARDS_JUMP; } else if ( !(pm->ps->forcePowersActive&(1<cmd.forwardmove >= 0 ) { if(!PM_InDeathAnim()) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_JUMP1,SETANIM_FLAG_OVERRIDE, 100); // Only blend over 100ms } pm->ps->pm_flags &= ~PMF_BACKWARDS_JUMP; } else if ( pm->cmd.forwardmove < 0 ) { if(!PM_InDeathAnim()) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_JUMPBACK1,SETANIM_FLAG_OVERRIDE, 100); // Only blend over 100ms } pm->ps->pm_flags |= PMF_BACKWARDS_JUMP; } } } else { // If we're a vehicle, set our falling flag. if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_VEHICLE ) { // We're on the ground. if ( pm->gent->m_pVehicle->m_pVehicleInfo->type == VH_ANIMAL ) { pm->gent->m_pVehicle->m_ulFlags &= ~VEH_FLYING; } } } } } } } if ( pm->ps->groundEntityNum != ENTITYNUM_NONE ) { pm->ps->jumpZStart = pm->ps->origin[2]; } } pm->ps->groundEntityNum = ENTITYNUM_NONE; pml.groundPlane = qfalse; pml.walking = qfalse; } /* ============= PM_GroundTrace ============= */ static void PM_GroundTrace( void ) { vec3_t point; trace_t trace; if ( (pm->ps->eFlags&EF_LOCKED_TO_WEAPON) || (pm->ps->eFlags&EF_HELD_BY_RANCOR) || (pm->ps->eFlags&EF_HELD_BY_WAMPA) || (pm->ps->eFlags&EF_HELD_BY_SAND_CREATURE) || PM_RidingVehicle() ) { pml.groundPlane = qtrue; pml.walking = qtrue; pm->ps->groundEntityNum = ENTITYNUM_WORLD; pm->ps->lastOnGround = level.time; /* pml.groundTrace.allsolid = qfalse; pml.groundTrace.contents = CONTENTS_SOLID; VectorCopy( pm->ps->origin, pml.groundTrace.endpos ); pml.groundTrace.entityNum = ENTITYNUM_WORLD; pml.groundTrace.fraction = 0.0f;; pml.groundTrace.G2CollisionMap = NULL; pml.groundTrace.plane.dist = 0.0f; VectorSet( pml.groundTrace.plane.normal, 0, 0, 1 ); pml.groundTrace.plane.pad = 0; pml.groundTrace.plane.signbits = 0; pml.groundTrace.plane.type = 0;; pml.groundTrace.startsolid = qfalse; pml.groundTrace.surfaceFlags = 0; */ return; } else if ( pm->ps->legsAnimTimer > 300 && (pm->ps->legsAnim == BOTH_WALL_RUN_RIGHT || pm->ps->legsAnim == BOTH_WALL_RUN_LEFT || pm->ps->legsAnim == BOTH_FORCEWALLRUNFLIP_START) ) {//wall-running forces you to be in the air pml.groundPlane = qfalse; pml.walking = qfalse; pm->ps->groundEntityNum = ENTITYNUM_NONE; return; } float minNormal = (float)MIN_WALK_NORMAL; if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_VEHICLE ) {//FIXME: extern this as part of vehicle data minNormal = pm->gent->m_pVehicle->m_pVehicleInfo->maxSlope; } point[0] = pm->ps->origin[0]; point[1] = pm->ps->origin[1]; point[2] = pm->ps->origin[2] - 0.25f; pm->trace (&trace, pm->ps->origin, pm->mins, pm->maxs, point, pm->ps->clientNum, pm->tracemask, (EG2_Collision)0, 0); pml.groundTrace = trace; // do something corrective if the trace starts in a solid... if ( trace.allsolid ) { PM_CorrectAllSolid(); return; } // if the trace didn't hit anything, we are in free fall if ( trace.fraction == 1.0 || g_gravity->value <= 0 ) { PM_GroundTraceMissed(); pml.groundPlane = qfalse; pml.walking = qfalse; /* if ( pm->ps->vehicleIndex != VEHICLE_NONE ) { PM_SetVehicleAngles( NULL ); } */ return; } // Not a vehicle and not riding one. if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class != CLASS_SAND_CREATURE && (pm->gent->client->NPC_class != CLASS_VEHICLE && !PM_RidingVehicle() ) ) { // check if getting thrown off the ground if ( ((pm->ps->velocity[2]>0&&(pm->ps->pm_flags&PMF_TIME_KNOCKBACK))||pm->ps->velocity[2]>100) && DotProduct( pm->ps->velocity, trace.plane.normal ) > 10 ) {//either thrown off ground (PMF_TIME_KNOCKBACK) or going off the ground at a large velocity if ( pm->debugLevel ) { Com_Printf("%i:kickoff\n", c_pmove); } // go into jump animation if ( PM_FlippingAnim( pm->ps->legsAnim) ) {//we're flipping } else if ( PM_InSpecialJump( pm->ps->legsAnim ) ) {//special jumps } else if ( PM_InKnockDown( pm->ps ) ) {//in knockdown } else if ( PM_InRoll( pm->ps ) ) {//in knockdown } else if ( PM_KickingAnim( pm->ps->legsAnim ) ) {//in kick } else if ( pm->gent && pm->gent->client && (pm->gent->client->NPC_class == CLASS_RANCOR || pm->gent->client->NPC_class == CLASS_WAMPA) ) { } else { PM_JumpForDir(); } pm->ps->groundEntityNum = ENTITYNUM_NONE; pml.groundPlane = qfalse; pml.walking = qfalse; return; } } /* if ( pm->ps->vehicleIndex != VEHICLE_NONE ) { PM_SetVehicleAngles( trace.plane.normal ); } */ // slopes that are too steep will not be considered onground if ( trace.plane.normal[2] < minNormal ) { if ( pm->debugLevel ) { Com_Printf("%i:steep\n", c_pmove); } // FIXME: if they can't slide down the slope, let them // walk (sharp crevices) pm->ps->groundEntityNum = ENTITYNUM_NONE; pml.groundPlane = qtrue; pml.walking = qfalse; return; } //FIXME: if the ground surface is a "cover surface (like tall grass), add a "cover" flag to me pml.groundPlane = qtrue; pml.walking = qtrue; // hitting solid ground will end a waterjump if (pm->ps->pm_flags & PMF_TIME_WATERJUMP) { pm->ps->pm_flags &= ~(PMF_TIME_WATERJUMP | PMF_TIME_LAND); pm->ps->pm_time = 0; } if ( pm->ps->groundEntityNum == ENTITYNUM_NONE ) { // just hit the ground if ( pm->debugLevel ) { Com_Printf("%i:Land\n", c_pmove); } //if ( !PM_ClientImpact( trace.entityNum, qtrue ) ) { PM_CrashLand(); // don't do landing time if we were just going down a slope if ( pml.previous_velocity[2] < -200 ) { // don't allow another jump for a little while pm->ps->pm_flags |= PMF_TIME_LAND; pm->ps->pm_time = 250; } if (!pm->cmd.forwardmove && !pm->cmd.rightmove) { if ( Flying != FLY_HOVER ) { pm->ps->velocity[2] = 0; //wouldn't normally want this because of slopes, but we aren't tyring to move... } } } } pm->ps->groundEntityNum = trace.entityNum; pm->ps->lastOnGround = level.time; if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) ) {//if a player, clear the jumping "flag" so can't double-jump pm->ps->forceJumpCharge = 0; } // don't reset the z velocity for slopes // pm->ps->velocity[2] = 0; PM_AddTouchEnt( trace.entityNum ); } int LastMatrixJumpTime = 0; #define DEBUGMATRIXJUMP 0 void PM_HoverTrace( void ) { if ( !pm->gent || !pm->gent->client || pm->gent->client->NPC_class != CLASS_VEHICLE ) { return; } Vehicle_t *pVeh = pm->gent->m_pVehicle; float hoverHeight = pVeh->m_pVehicleInfo->hoverHeight; vec3_t point, vAng, fxAxis[3]; trace_t *trace = &pml.groundTrace; int traceContents = pm->tracemask; pml.groundPlane = qfalse; float relativeWaterLevel = (pm->ps->waterheight - (pm->ps->origin[2]+pm->mins[2])); if ( pm->waterlevel && relativeWaterLevel >= 0 ) {//in water if ( pVeh->m_pVehicleInfo->bouyancy <= 0.0f ) {//sink like a rock } else {//rise up float floatHeight = (pVeh->m_pVehicleInfo->bouyancy * ((pm->maxs[2]-pm->mins[2])*0.5f)) - (hoverHeight*0.5f);//1.0f should make you float half-in, half-out of water if ( relativeWaterLevel > floatHeight ) {//too low, should rise up pm->ps->velocity[2] += (relativeWaterLevel - floatHeight) * pVeh->m_fTimeModifier; } } if ( pm->ps->waterheight < pm->ps->origin[2]+pm->maxs[2] ) {//part of us is sticking out of water if ( fabs(pm->ps->velocity[0]) + fabs(pm->ps->velocity[1]) > 100 ) {//moving at a decent speed if ( Q_irand( pml.frametime, 100 ) >= 50 ) {//splash vAng[PITCH] = vAng[ROLL] = 0; vAng[YAW] = pVeh->m_vOrientation[YAW]; AngleVectors( vAng, fxAxis[2], fxAxis[1], fxAxis[0] ); vec3_t wakeOrg; VectorCopy( pm->ps->origin, wakeOrg ); wakeOrg[2] = pm->ps->waterheight; if ( pVeh->m_pVehicleInfo->iWakeFX ) { G_PlayEffect( pVeh->m_pVehicleInfo->iWakeFX, wakeOrg, fxAxis ); } } } pml.groundPlane = qtrue; } } else { float minNormal = (float)MIN_WALK_NORMAL; minNormal = pVeh->m_pVehicleInfo->maxSlope; point[0] = pm->ps->origin[0]; point[1] = pm->ps->origin[1]; point[2] = pm->ps->origin[2] - (hoverHeight*3.0f); //FIXME: check for water, too? If over water, go slower and make wave effect // If *in* water, go really slow and use bouyancy stat to determine how far below surface to float //NOTE: if bouyancy is 2.0f or higher, you float over water like it's solid ground. // if it's 1.0f, you sink halfway into water. If it's 0, you sink... if ( pVeh->m_pVehicleInfo->bouyancy >= 2.0f ) {//sit on water traceContents |= (CONTENTS_WATER|CONTENTS_SLIME|CONTENTS_LAVA); } pm->trace( trace, pm->ps->origin, pm->mins, pm->maxs, point, pm->ps->clientNum, traceContents, (EG2_Collision)0, 0 ); if ( trace->plane.normal[2] >= minNormal ) {//not a steep slope, so push us up if ( trace->fraction < 0.3f ) {//push up off ground float hoverForce = pVeh->m_pVehicleInfo->hoverStrength; pm->ps->velocity[2] += (0.3f-trace->fraction)*hoverForce*pVeh->m_fTimeModifier; // if (pm->ps->velocity[2]>60.0f) // { // pm->ps->velocity[2] = 60.0f; // } if ( (trace->contents&(CONTENTS_WATER|CONTENTS_SLIME|CONTENTS_LAVA)) ) {//hovering on water, make a spash if moving if ( fabs(pm->ps->velocity[0]) + fabs(pm->ps->velocity[1]) > 100 ) {//moving at a decent speed if ( Q_irand( pml.frametime, 100 ) >= 50 ) {//splash vAng[PITCH] = vAng[ROLL] = 0; vAng[YAW] = pVeh->m_vOrientation[YAW]; AngleVectors( vAng, fxAxis[2], fxAxis[1], fxAxis[0] ); if ( pVeh->m_pVehicleInfo->iWakeFX ) { G_PlayEffect( pVeh->m_pVehicleInfo->iWakeFX, trace->endpos, fxAxis ); } } } } if (pVeh->m_ulFlags & VEH_SLIDEBREAKING) { if ( Q_irand( pml.frametime, 100 ) >= 50 ) {//dust VectorClear(fxAxis[0]); fxAxis[0][2] = 1; VectorCopy(pm->ps->velocity, fxAxis[1]); fxAxis[1][2] *= 0.01f; VectorMA(pm->ps->origin, 0.25f, fxAxis[1], point); G_PlayEffect("ships/swoop_dust", point, fxAxis[0]); } } pml.groundPlane = qtrue; } } } if ( pml.groundPlane ) { PM_SetVehicleAngles( pml.groundTrace.plane.normal ); // We're on the ground. // if (pVeh->m_ulFlags&VEH_FLYING && level.timem_iTurboTime) // { // pVeh->m_iTurboTime = 0; // stop turbo // } pVeh->m_ulFlags &= ~VEH_FLYING; pVeh->m_vAngularVelocity = 0.0f; } else { PM_SetVehicleAngles( NULL ); // We're flying in the air. pVeh->m_ulFlags |= VEH_FLYING; //groundTrace if (pVeh->m_vAngularVelocity==0.0f) { pVeh->m_vAngularVelocity = pVeh->m_vOrientation[YAW] - pVeh->m_vPrevOrientation[YAW]; if (pVeh->m_vAngularVelocity<-15.0f) { pVeh->m_vAngularVelocity = -15.0f; } if (pVeh->m_vAngularVelocity> 15.0f) { pVeh->m_vAngularVelocity = 15.0f; } // BEGIN MATRIX MODE INIT FOR JUMP //================================= if (pm->gent && pm->gent->owner && (pm->gent->owner->s.numbergent->owner)) && pVeh->m_pVehicleInfo->type==VH_SPEEDER && level.time>(LastMatrixJumpTime+5000) && VectorLength(pm->ps->velocity)>30.0f) { LastMatrixJumpTime = level.time; vec3_t predictedApx; vec3_t predictedFallVelocity; vec3_t predictedLandPosition; VectorScale(pm->ps->velocity, 2.0f, predictedFallVelocity); // take friction into account predictedFallVelocity[2] = -(pm->ps->gravity * 1.1f); // take gravity into account VectorMA(pm->ps->origin, 0.25f, pm->ps->velocity, predictedApx); VectorMA(predictedApx, 0.25f, predictedFallVelocity, predictedLandPosition); trace_t trace2; gi.trace( &trace2, predictedApx, pm->mins, pm->maxs, predictedLandPosition, pm->ps->clientNum, traceContents, (EG2_Collision)0, 0); if (!trace2.startsolid && !trace2.allsolid && trace2.fraction>0.75 && Q_irand(0, 3)==0) { LastMatrixJumpTime += 20000; G_StartMatrixEffect(pm->gent, MEF_HIT_GROUND_STOP); // CG_DrawEdge(pm->ps->origin, predictedApx, EDGE_WHITE_TWOSECOND); // CG_DrawEdge(predictedApx, predictedLandPosition, EDGE_WHITE_TWOSECOND); } // else // { // CG_DrawEdge(pm->ps->origin, predictedApx, EDGE_RED_TWOSECOND); // CG_DrawEdge(predictedApx, predictedLandPosition, EDGE_RED_TWOSECOND); // } } //================================= } pVeh->m_vAngularVelocity *= 0.95f; // Angular Velocity Decays Over Time } PM_GroundTraceMissed(); } /* ============= PM_SetWaterLevelAtPoint FIXME: avoid this twice? certainly if not moving ============= */ static void PM_SetWaterLevelAtPoint( vec3_t org, int *waterlevel, int *watertype ) { vec3_t point; int cont; int sample1; int sample2; // // get waterlevel, accounting for ducking // *waterlevel = 0; *watertype = 0; point[0] = org[0]; point[1] = org[1]; point[2] = org[2] + DEFAULT_MINS_2 + 1; if (gi.totalMapContents() & (MASK_WATER|CONTENTS_LADDER)) { cont = pm->pointcontents( point, pm->ps->clientNum ); if ( cont & (MASK_WATER|CONTENTS_LADDER) ) { sample2 = pm->ps->viewheight - DEFAULT_MINS_2; sample1 = sample2 / 2; *watertype = cont; *waterlevel = 1; point[2] = org[2] + DEFAULT_MINS_2 + sample1; cont = pm->pointcontents( point, pm->ps->clientNum ); if ( cont & (MASK_WATER|CONTENTS_LADDER) ) { *waterlevel = 2; point[2] = org[2] + DEFAULT_MINS_2 + sample2; cont = pm->pointcontents( point, pm->ps->clientNum ); if ( cont & (MASK_WATER|CONTENTS_LADDER) ) { *waterlevel = 3; } } } } } void PM_SetWaterHeight( void ) { pm->ps->waterHeightLevel = WHL_NONE; if ( pm->waterlevel < 1 ) { pm->ps->waterheight = pm->ps->origin[2] + DEFAULT_MINS_2 - 4; return; } trace_t trace; vec3_t top, bottom; VectorCopy( pm->ps->origin, top ); VectorCopy( pm->ps->origin, bottom ); top[2] += pm->gent->client->standheight; bottom[2] += DEFAULT_MINS_2; gi.trace( &trace, top, pm->mins, pm->maxs, bottom, pm->ps->clientNum, MASK_WATER, (EG2_Collision)0, 0 ); if ( trace.startsolid ) {//under water pm->ps->waterheight = top[2] + 4; } else if ( trace.fraction < 1.0f ) {//partially in and partially out of water pm->ps->waterheight = trace.endpos[2]+pm->mins[2]; } else if ( trace.contents&MASK_WATER ) {//water is above me pm->ps->waterheight = top[2] + 4; } else {//water is below me pm->ps->waterheight = bottom[2] - 4; } float distFromEyes = (pm->ps->origin[2]+pm->gent->client->standheight)-pm->ps->waterheight; if ( distFromEyes < 0 ) { pm->ps->waterHeightLevel = WHL_UNDER; } else if ( distFromEyes < 6 ) { pm->ps->waterHeightLevel = WHL_HEAD; } else if ( distFromEyes < 18 ) { pm->ps->waterHeightLevel = WHL_SHOULDERS; } else if ( distFromEyes < pm->gent->client->standheight-8 ) {//at least 8 above origin pm->ps->waterHeightLevel = WHL_TORSO; } else { float distFromOrg = pm->ps->origin[2]-pm->ps->waterheight; if ( distFromOrg < 6 ) { pm->ps->waterHeightLevel = WHL_WAIST; } else if ( distFromOrg < 16 ) { pm->ps->waterHeightLevel = WHL_KNEES; } else if ( distFromOrg > fabs(pm->mins[2]) ) { pm->ps->waterHeightLevel = WHL_NONE; } else { pm->ps->waterHeightLevel = WHL_ANKLES; } } } /* ============== PM_SetBounds Sets mins, maxs ============== */ static void PM_SetBounds (void) { if ( pm->gent && pm->gent->client ) { if ( !pm->gent->mins[0] || !pm->gent->mins[1] || !pm->gent->mins[2] || !pm->gent->maxs[0] || !pm->gent->maxs[1] || !pm->gent->maxs[2] ) { //assert(0); } VectorCopy( pm->gent->mins, pm->mins ); VectorCopy( pm->gent->maxs, pm->maxs ); } else { if ( !DEFAULT_MINS_0 || !DEFAULT_MINS_1 || !DEFAULT_MAXS_0 || !DEFAULT_MAXS_1 || !DEFAULT_MINS_2 || !DEFAULT_MAXS_2 ) { assert(0); } pm->mins[0] = DEFAULT_MINS_0; pm->mins[1] = DEFAULT_MINS_1; pm->maxs[0] = DEFAULT_MAXS_0; pm->maxs[1] = DEFAULT_MAXS_1; pm->mins[2] = DEFAULT_MINS_2; pm->maxs[2] = DEFAULT_MAXS_2; } } /* ============== PM_CheckDuck Sets mins, maxs, and pm->ps->viewheight ============== */ static void PM_CheckDuck (void) { trace_t trace; int standheight; int crouchheight; int oldHeight; if ( pm->gent && pm->gent->client ) { if ( !pm->gent->mins[0] || !pm->gent->mins[1] || !pm->gent->mins[2] || !pm->gent->maxs[0] || !pm->gent->maxs[1] || !pm->gent->maxs[2] ) { //assert(0); } if ( pm->ps->clientNum < MAX_CLIENTS && (pm->gent->client->NPC_class == CLASS_ATST ||pm->gent->client->NPC_class == CLASS_RANCOR) && !cg.renderingThirdPerson ) { standheight = crouchheight = 128; } else { standheight = pm->gent->client->standheight; crouchheight = pm->gent->client->crouchheight; } } else { if ( !DEFAULT_MINS_0 || !DEFAULT_MINS_1 || !DEFAULT_MAXS_0 || !DEFAULT_MAXS_1 || !DEFAULT_MINS_2 || !DEFAULT_MAXS_2 ) { assert(0); } standheight = DEFAULT_MAXS_2; crouchheight = CROUCH_MAXS_2; } if ( PM_RidingVehicle() || (pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_VEHICLE) ) {//riding a vehicle or are a vehicle //no ducking or rolling when on a vehicle //right? not even on ones that you just ride on top of? pm->ps->pm_flags &= ~PMF_DUCKED; pm->maxs[2] = standheight; //FIXME: have a crouchviewheight and standviewheight on ent? pm->ps->viewheight = standheight + STANDARD_VIEWHEIGHT_OFFSET;//DEFAULT_VIEWHEIGHT; //NOTE: we don't clear the pm->cmd.upmove here because //the vehicle code may need it later... but, for riders, //it should have already been copied over to the vehicle, right? return; } if ( PM_InGetUp( pm->ps ) ) {//can't do any kind of crouching when getting up if ( pm->ps->legsAnim == BOTH_GETUP_CROUCH_B1 || pm->ps->legsAnim == BOTH_GETUP_CROUCH_F1 ) {//crouched still pm->ps->pm_flags |= PMF_DUCKED; pm->maxs[2] = crouchheight; } pm->ps->viewheight = crouchheight + STANDARD_VIEWHEIGHT_OFFSET; return; } oldHeight = pm->maxs[2]; if ( PM_InRoll( pm->ps ) ) { /* if ( pm->ps->clientNum && pm->gent && pm->gent->client ) { pm->maxs[2] = pm->gent->client->renderInfo.eyePoint[2]-pm->ps->origin[2] + 4; if ( crouchheight > pm->maxs[2] ) { pm->maxs[2] = crouchheight; } } else */ { pm->maxs[2] = crouchheight; } pm->ps->viewheight = crouchheight + STANDARD_VIEWHEIGHT_OFFSET; pm->ps->pm_flags |= PMF_DUCKED; return; } if ( PM_GettingUpFromKnockDown( standheight, crouchheight ) ) { pm->ps->viewheight = crouchheight + STANDARD_VIEWHEIGHT_OFFSET; return; } if ( PM_InKnockDown( pm->ps ) ) {//forced crouch if ( pm->gent && pm->gent->client ) {//interrupted any potential delayed weapon fires pm->gent->client->fireDelay = 0; } pm->maxs[2] = crouchheight; pm->ps->viewheight = crouchheight + STANDARD_VIEWHEIGHT_OFFSET; pm->ps->pm_flags |= PMF_DUCKED; return; } if ( pm->cmd.upmove < 0 ) { // trying to duck pm->maxs[2] = crouchheight; pm->ps->viewheight = crouchheight + STANDARD_VIEWHEIGHT_OFFSET;//CROUCH_VIEWHEIGHT; if ( pm->ps->groundEntityNum == ENTITYNUM_NONE && !PM_SwimmingAnim( pm->ps->legsAnim ) ) {//Not ducked already and trying to duck in mid-air //will raise your feet, unducking whilst in air will drop feet if ( !(pm->ps->pm_flags&PMF_DUCKED) ) { pm->ps->eFlags ^= EF_TELEPORT_BIT; } if ( pm->gent ) { pm->ps->origin[2] += oldHeight - pm->maxs[2];//diff will be zero if were already ducking //Don't worry, we know we fit in a smaller size } } pm->ps->pm_flags |= PMF_DUCKED; if ( d_JediAI->integer ) { if ( pm->ps->clientNum && pm->ps->weapon == WP_SABER ) { Com_Printf( "ducking\n" ); } } } else { // want to stop ducking, stand up if possible if ( pm->ps->pm_flags & PMF_DUCKED ) {//Was ducking if ( pm->ps->groundEntityNum == ENTITYNUM_NONE ) {//unducking whilst in air will try to drop feet pm->maxs[2] = standheight; pm->ps->origin[2] += oldHeight - pm->maxs[2]; pm->trace (&trace, pm->ps->origin, pm->mins, pm->maxs, pm->ps->origin, pm->ps->clientNum, pm->tracemask, (EG2_Collision)0, 0 ); if ( !trace.allsolid ) { pm->ps->eFlags ^= EF_TELEPORT_BIT; pm->ps->pm_flags &= ~PMF_DUCKED; } else {//Put us back pm->ps->origin[2] -= oldHeight - pm->maxs[2]; } //NOTE: this isn't the best way to check this, you may have room to unduck //while in air, but your feet are close to landing. Probably won't be a //noticable shortcoming } else { // try to stand up pm->maxs[2] = standheight; pm->trace( &trace, pm->ps->origin, pm->mins, pm->maxs, pm->ps->origin, pm->ps->clientNum, pm->tracemask, (EG2_Collision)0, 0 ); if ( !trace.allsolid ) { pm->ps->pm_flags &= ~PMF_DUCKED; } } } if ( pm->ps->pm_flags & PMF_DUCKED ) {//Still ducking pm->maxs[2] = crouchheight; pm->ps->viewheight = crouchheight + STANDARD_VIEWHEIGHT_OFFSET;//CROUCH_VIEWHEIGHT; } else {//standing now pm->maxs[2] = standheight; //FIXME: have a crouchviewheight and standviewheight on ent? pm->ps->viewheight = standheight + STANDARD_VIEWHEIGHT_OFFSET;//DEFAULT_VIEWHEIGHT; } } } //=================================================================== qboolean PM_SaberLockAnim( int anim ) { switch ( anim ) { case BOTH_BF2LOCK: //# case BOTH_BF1LOCK: //# case BOTH_CWCIRCLELOCK: //# case BOTH_CCWCIRCLELOCK: //# return qtrue; } return qfalse; } qboolean PM_ForceAnim( int anim ) { switch ( anim ) { case BOTH_CHOKE1: //being choked...??? case BOTH_GESTURE1: //taunting... case BOTH_RESISTPUSH: //# plant yourself to resist force push/pulls. case BOTH_FORCEPUSH: //# Use off-hand to do force power. case BOTH_FORCEPULL: //# Use off-hand to do force power. case BOTH_MINDTRICK1: //# Use off-hand to do mind trick case BOTH_MINDTRICK2: //# Use off-hand to do distraction case BOTH_FORCELIGHTNING: //# Use off-hand to do lightning case BOTH_FORCELIGHTNING_START: case BOTH_FORCELIGHTNING_HOLD: //# Use off-hand to do lightning case BOTH_FORCELIGHTNING_RELEASE: //# Use off-hand to do lightning case BOTH_FORCEHEAL_START: //# Healing meditation pose start case BOTH_FORCEHEAL_STOP: //# Healing meditation pose end case BOTH_FORCEHEAL_QUICK: //# Healing meditation gesture case BOTH_FORCEGRIP1: //# temp force-grip anim (actually re-using push) case BOTH_FORCEGRIP_HOLD: //# temp force-grip anim (actually re-using push) case BOTH_FORCEGRIP_RELEASE: //# temp force-grip anim (actually re-using push) //case BOTH_FORCEGRIP3: //# force-gripping case BOTH_FORCE_RAGE: case BOTH_FORCE_2HANDEDLIGHTNING: case BOTH_FORCE_2HANDEDLIGHTNING_START: case BOTH_FORCE_2HANDEDLIGHTNING_HOLD: case BOTH_FORCE_2HANDEDLIGHTNING_RELEASE: case BOTH_FORCE_DRAIN: case BOTH_FORCE_DRAIN_START: case BOTH_FORCE_DRAIN_HOLD: case BOTH_FORCE_DRAIN_RELEASE: case BOTH_FORCE_ABSORB: case BOTH_FORCE_ABSORB_START: case BOTH_FORCE_ABSORB_END: case BOTH_FORCE_PROTECT: case BOTH_FORCE_PROTECT_FAST: return qtrue; break; } return qfalse; } qboolean PM_InSaberAnim( int anim ) { if ( anim >= BOTH_A1_T__B_ && anim <= BOTH_H1_S1_BR ) { return qtrue; } return qfalse; } qboolean PM_InForceGetUp( playerState_t *ps ) { switch ( ps->legsAnim ) { case BOTH_FORCE_GETUP_F1: case BOTH_FORCE_GETUP_F2: case BOTH_FORCE_GETUP_B1: case BOTH_FORCE_GETUP_B2: case BOTH_FORCE_GETUP_B3: case BOTH_FORCE_GETUP_B4: case BOTH_FORCE_GETUP_B5: case BOTH_FORCE_GETUP_B6: case BOTH_GETUP_BROLL_B: case BOTH_GETUP_BROLL_F: case BOTH_GETUP_BROLL_L: case BOTH_GETUP_BROLL_R: case BOTH_GETUP_FROLL_B: case BOTH_GETUP_FROLL_F: case BOTH_GETUP_FROLL_L: case BOTH_GETUP_FROLL_R: if ( ps->legsAnimTimer ) { return qtrue; } break; } return qfalse; } qboolean PM_InGetUp( playerState_t *ps ) { switch ( ps->legsAnim ) { case BOTH_GETUP1: case BOTH_GETUP2: case BOTH_GETUP3: case BOTH_GETUP4: case BOTH_GETUP5: case BOTH_GETUP_CROUCH_F1: case BOTH_GETUP_CROUCH_B1: case BOTH_GETUP_BROLL_B: case BOTH_GETUP_BROLL_F: case BOTH_GETUP_BROLL_L: case BOTH_GETUP_BROLL_R: case BOTH_GETUP_FROLL_B: case BOTH_GETUP_FROLL_F: case BOTH_GETUP_FROLL_L: case BOTH_GETUP_FROLL_R: if ( ps->legsAnimTimer ) { return qtrue; } break; default: return PM_InForceGetUp( ps ); break; } //what the hell, redundant, but... return qfalse; } qboolean PM_InGetUpNoRoll( playerState_t *ps ) { switch ( ps->legsAnim ) { case BOTH_GETUP1: case BOTH_GETUP2: case BOTH_GETUP3: case BOTH_GETUP4: case BOTH_GETUP5: case BOTH_GETUP_CROUCH_F1: case BOTH_GETUP_CROUCH_B1: case BOTH_FORCE_GETUP_F1: case BOTH_FORCE_GETUP_F2: case BOTH_FORCE_GETUP_B1: case BOTH_FORCE_GETUP_B2: case BOTH_FORCE_GETUP_B3: case BOTH_FORCE_GETUP_B4: case BOTH_FORCE_GETUP_B5: case BOTH_FORCE_GETUP_B6: if ( ps->legsAnimTimer ) { return qtrue; } break; } return qfalse; } qboolean PM_InKnockDown( playerState_t *ps ) { switch ( ps->legsAnim ) { case BOTH_KNOCKDOWN1: case BOTH_KNOCKDOWN2: case BOTH_KNOCKDOWN3: case BOTH_KNOCKDOWN4: case BOTH_KNOCKDOWN5: //special anims: case BOTH_RELEASED: return qtrue; break; case BOTH_LK_DL_ST_T_SB_1_L: if ( ps->legsAnimTimer < 550 ) { return qtrue; } break; case BOTH_PLAYER_PA_3_FLY: if ( ps->legsAnimTimer < 300 ) { return qtrue; } /* else if ( ps->clientNum < MAX_CLIENTS && ps->legsAnimTimer < 300 + PLAYER_KNOCKDOWN_HOLD_EXTRA_TIME ) { return qtrue; } */ break; default: return PM_InGetUp( ps ); break; } return qfalse; } qboolean PM_InKnockDownNoGetup( playerState_t *ps ) { switch ( ps->legsAnim ) { case BOTH_KNOCKDOWN1: case BOTH_KNOCKDOWN2: case BOTH_KNOCKDOWN3: case BOTH_KNOCKDOWN4: case BOTH_KNOCKDOWN5: //special anims: case BOTH_RELEASED: return qtrue; break; case BOTH_LK_DL_ST_T_SB_1_L: if ( ps->legsAnimTimer < 550 ) { return qtrue; } break; case BOTH_PLAYER_PA_3_FLY: if ( ps->legsAnimTimer < 300 ) { return qtrue; } /* else if ( ps->clientNum < MAX_CLIENTS && ps->legsAnimTimer < 300 + PLAYER_KNOCKDOWN_HOLD_EXTRA_TIME ) { return qtrue; } */ break; } return qfalse; } qboolean PM_InKnockDownOnGround( playerState_t *ps ) { switch ( ps->legsAnim ) { case BOTH_KNOCKDOWN1: case BOTH_KNOCKDOWN2: case BOTH_KNOCKDOWN3: case BOTH_KNOCKDOWN4: case BOTH_KNOCKDOWN5: case BOTH_RELEASED: //if ( PM_AnimLength( g_entities[ps->clientNum].client->clientInfo.animFileIndex, (animNumber_t)ps->legsAnim ) - ps->legsAnimTimer > 300 ) {//at end of fall down anim return qtrue; } break; case BOTH_GETUP1: case BOTH_GETUP2: case BOTH_GETUP3: case BOTH_GETUP4: case BOTH_GETUP5: case BOTH_GETUP_CROUCH_F1: case BOTH_GETUP_CROUCH_B1: case BOTH_FORCE_GETUP_F1: case BOTH_FORCE_GETUP_F2: case BOTH_FORCE_GETUP_B1: case BOTH_FORCE_GETUP_B2: case BOTH_FORCE_GETUP_B3: case BOTH_FORCE_GETUP_B4: case BOTH_FORCE_GETUP_B5: case BOTH_FORCE_GETUP_B6: if ( PM_AnimLength( g_entities[ps->clientNum].client->clientInfo.animFileIndex, (animNumber_t)ps->legsAnim ) - ps->legsAnimTimer < 500 ) {//at beginning of getup anim return qtrue; } break; case BOTH_GETUP_BROLL_B: case BOTH_GETUP_BROLL_F: case BOTH_GETUP_BROLL_L: case BOTH_GETUP_BROLL_R: case BOTH_GETUP_FROLL_B: case BOTH_GETUP_FROLL_F: case BOTH_GETUP_FROLL_L: case BOTH_GETUP_FROLL_R: if ( PM_AnimLength( g_entities[ps->clientNum].client->clientInfo.animFileIndex, (animNumber_t)ps->legsAnim ) - ps->legsAnimTimer < 500 ) {//at beginning of getup anim return qtrue; } break; case BOTH_LK_DL_ST_T_SB_1_L: if ( ps->legsAnimTimer < 1000 ) { return qtrue; } break; case BOTH_PLAYER_PA_3_FLY: if ( ps->legsAnimTimer < 300 ) { return qtrue; } /* else if ( ps->clientNum < MAX_CLIENTS && ps->legsAnimTimer < 300 + PLAYER_KNOCKDOWN_HOLD_EXTRA_TIME ) { return qtrue; } */ break; } return qfalse; } qboolean PM_CrouchGetup( float crouchheight ) { pm->maxs[2] = crouchheight; pm->ps->viewheight = crouchheight + STANDARD_VIEWHEIGHT_OFFSET; int anim = -1; switch ( pm->ps->legsAnim ) { case BOTH_KNOCKDOWN1: case BOTH_KNOCKDOWN2: case BOTH_KNOCKDOWN4: case BOTH_RELEASED: case BOTH_PLAYER_PA_3_FLY: anim = BOTH_GETUP_CROUCH_B1; break; case BOTH_KNOCKDOWN3: case BOTH_KNOCKDOWN5: case BOTH_LK_DL_ST_T_SB_1_L: anim = BOTH_GETUP_CROUCH_F1; break; } if ( anim == -1 ) {//WTF? stay down? pm->ps->legsAnimTimer = 100;//hold this anim for another 10th of a second return qfalse; } else {//get up into crouch anim if ( PM_LockedAnim( pm->ps->torsoAnim ) ) {//need to be able to override this anim pm->ps->torsoAnimTimer = 0; } if ( PM_LockedAnim( pm->ps->legsAnim ) ) {//need to be able to override this anim pm->ps->legsAnimTimer = 0; } PM_SetAnim( pm, SETANIM_BOTH, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD|SETANIM_FLAG_HOLDLESS ); pm->ps->saberMove = pm->ps->saberBounceMove = LS_READY;//don't finish whatever saber anim you may have been in pm->ps->saberBlocked = BLOCKED_NONE; return qtrue; } } extern qboolean PM_GoingToAttackDown( playerState_t *ps ); qboolean PM_CheckRollGetup( void ) { if ( pm->ps->legsAnim == BOTH_KNOCKDOWN1 || pm->ps->legsAnim == BOTH_KNOCKDOWN2 || pm->ps->legsAnim == BOTH_KNOCKDOWN3 || pm->ps->legsAnim == BOTH_KNOCKDOWN4 || pm->ps->legsAnim == BOTH_KNOCKDOWN5 || pm->ps->legsAnim == BOTH_LK_DL_ST_T_SB_1_L || pm->ps->legsAnim == BOTH_PLAYER_PA_3_FLY || pm->ps->legsAnim == BOTH_RELEASED ) {//lying on back or front if ( ((pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && ( pm->cmd.rightmove || (pm->cmd.forwardmove&&pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_0) ) )//player pressing left or right || ( (pm->ps->clientNum >= MAX_CLIENTS&&!PM_ControlledByPlayer()) && pm->gent->NPC//an NPC && pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_0//have at least force jump 1 && pm->gent->enemy //I have an enemy && pm->gent->enemy->client//a client && pm->gent->enemy->enemy == pm->gent//he's mad at me! && (PM_GoingToAttackDown( &pm->gent->enemy->client->ps )||!Q_irand(0,2))//he's attacking downward! (or we just feel like doing it this time) && ((pm->gent->client&&pm->gent->client->NPC_class==CLASS_ALORA)||Q_irand( 0, RANK_CAPTAIN )gent->NPC->rank) ) )//higher rank I am, more likely I am to roll away! {//roll away! int anim; qboolean forceGetUp = qfalse; if ( pm->cmd.forwardmove > 0 ) { if ( pm->ps->legsAnim == BOTH_KNOCKDOWN3 || pm->ps->legsAnim == BOTH_KNOCKDOWN5 || pm->ps->legsAnim == BOTH_LK_DL_ST_T_SB_1_L ) { anim = BOTH_GETUP_FROLL_F; } else { anim = BOTH_GETUP_BROLL_F; } forceGetUp = qtrue; } else if ( pm->cmd.forwardmove < 0 ) { if ( pm->ps->legsAnim == BOTH_KNOCKDOWN3 || pm->ps->legsAnim == BOTH_KNOCKDOWN5 || pm->ps->legsAnim == BOTH_LK_DL_ST_T_SB_1_L ) { anim = BOTH_GETUP_FROLL_B; } else { anim = BOTH_GETUP_BROLL_B; } forceGetUp = qtrue; } else if ( pm->cmd.rightmove > 0 ) { if ( pm->ps->legsAnim == BOTH_KNOCKDOWN3 || pm->ps->legsAnim == BOTH_KNOCKDOWN5 || pm->ps->legsAnim == BOTH_LK_DL_ST_T_SB_1_L ) { anim = BOTH_GETUP_FROLL_R; } else { anim = BOTH_GETUP_BROLL_R; } } else if ( pm->cmd.rightmove < 0 ) { if ( pm->ps->legsAnim == BOTH_KNOCKDOWN3 || pm->ps->legsAnim == BOTH_KNOCKDOWN5 || pm->ps->legsAnim == BOTH_LK_DL_ST_T_SB_1_L ) { anim = BOTH_GETUP_FROLL_L; } else { anim = BOTH_GETUP_BROLL_L; } } else { if ( pm->ps->legsAnim == BOTH_KNOCKDOWN3 || pm->ps->legsAnim == BOTH_KNOCKDOWN5 || pm->ps->legsAnim == BOTH_LK_DL_ST_T_SB_1_L ) {//on your front anim = Q_irand( BOTH_GETUP_FROLL_B, BOTH_GETUP_FROLL_R ); } else { anim = Q_irand( BOTH_GETUP_BROLL_B, BOTH_GETUP_BROLL_R ); } } if ( (pm->ps->clientNum >= MAX_CLIENTS&&!PM_ControlledByPlayer()) ) { if ( !G_CheckRollSafety( pm->gent, anim, 64 ) ) {//oops, try other one if ( pm->ps->legsAnim == BOTH_KNOCKDOWN3 || pm->ps->legsAnim == BOTH_KNOCKDOWN5 || pm->ps->legsAnim == BOTH_LK_DL_ST_T_SB_1_L ) { if ( anim == BOTH_GETUP_FROLL_R ) { anim = BOTH_GETUP_FROLL_L; } else if ( anim == BOTH_GETUP_FROLL_F ) { anim = BOTH_GETUP_FROLL_B; } else if ( anim == BOTH_GETUP_FROLL_B ) { anim = BOTH_GETUP_FROLL_F; } else { anim = BOTH_GETUP_FROLL_L; } if ( !G_CheckRollSafety( pm->gent, anim, 64 ) ) {//neither side is clear, screw it return qfalse; } } else { if ( anim == BOTH_GETUP_BROLL_R ) { anim = BOTH_GETUP_BROLL_L; } else if ( anim == BOTH_GETUP_BROLL_F ) { anim = BOTH_GETUP_BROLL_B; } else if ( anim == BOTH_GETUP_FROLL_B ) { anim = BOTH_GETUP_BROLL_F; } else { anim = BOTH_GETUP_BROLL_L; } if ( !G_CheckRollSafety( pm->gent, anim, 64 ) ) {//neither side is clear, screw it return qfalse; } } } } pm->cmd.rightmove = pm->cmd.forwardmove = 0; if ( PM_LockedAnim( pm->ps->torsoAnim ) ) {//need to be able to override this anim pm->ps->torsoAnimTimer = 0; } if ( PM_LockedAnim( pm->ps->legsAnim ) ) {//need to be able to override this anim pm->ps->legsAnimTimer = 0; } PM_SetAnim( pm, SETANIM_BOTH, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD|SETANIM_FLAG_HOLDLESS ); pm->ps->weaponTime = pm->ps->torsoAnimTimer - 300;//don't attack until near end of this anim pm->ps->saberMove = pm->ps->saberBounceMove = LS_READY;//don't finish whatever saber anim you may have been in pm->ps->saberBlocked = BLOCKED_NONE; if ( forceGetUp ) { if ( pm->gent && pm->gent->client && pm->gent->client->playerTeam == TEAM_ENEMY && pm->gent->NPC && pm->gent->NPC->blockedSpeechDebounceTime < level.time && !Q_irand( 0, 1 ) ) { PM_AddEvent( Q_irand( EV_COMBAT1, EV_COMBAT3 ) ); pm->gent->NPC->blockedSpeechDebounceTime = level.time + 1000; } G_SoundOnEnt( pm->gent, CHAN_BODY, "sound/weapons/force/jump.wav" ); //launch off ground? pm->ps->weaponTime = 300;//just to make sure it's cleared } return qtrue; } } return qfalse; } extern int G_MinGetUpTime( gentity_t *ent ); qboolean PM_GettingUpFromKnockDown( float standheight, float crouchheight ) { int legsAnim = pm->ps->legsAnim; if ( legsAnim == BOTH_KNOCKDOWN1 ||legsAnim == BOTH_KNOCKDOWN2 ||legsAnim == BOTH_KNOCKDOWN3 ||legsAnim == BOTH_KNOCKDOWN4 ||legsAnim == BOTH_KNOCKDOWN5 ||legsAnim == BOTH_PLAYER_PA_3_FLY ||legsAnim == BOTH_LK_DL_ST_T_SB_1_L ||legsAnim == BOTH_RELEASED ) {//in a knockdown int minTimeLeft = G_MinGetUpTime( pm->gent ); if ( pm->ps->legsAnimTimer <= minTimeLeft ) {//if only a quarter of a second left, allow roll-aways if ( PM_CheckRollGetup() ) { pm->cmd.rightmove = pm->cmd.forwardmove = 0; return qtrue; } } if ( TIMER_Exists( pm->gent, "noGetUpStraight" ) ) { if ( !TIMER_Done2( pm->gent, "noGetUpStraight", qtrue ) ) {//not allowed to do straight get-ups for another few seconds if ( pm->ps->legsAnimTimer <= minTimeLeft ) {//hold it for a bit pm->ps->legsAnimTimer = minTimeLeft+1; } } } if ( !pm->ps->legsAnimTimer || (pm->ps->legsAnimTimer<=minTimeLeft&&(pm->cmd.upmove>0||(pm->gent&&pm->gent->client&&pm->gent->client->NPC_class==CLASS_ALORA))) ) {//done with the knockdown - FIXME: somehow this is allowing an *instant* getup...??? //FIXME: if trying to crouch (holding button?), just get up into a crouch? if ( pm->cmd.upmove < 0 ) { return PM_CrouchGetup( crouchheight ); } else { trace_t trace; // try to stand up pm->maxs[2] = standheight; pm->trace( &trace, pm->ps->origin, pm->mins, pm->maxs, pm->ps->origin, pm->ps->clientNum, pm->tracemask, (EG2_Collision)0, 0 ); if ( !trace.allsolid ) {//stand up int anim = BOTH_GETUP1; qboolean forceGetUp = qfalse; pm->maxs[2] = standheight; pm->ps->viewheight = standheight + STANDARD_VIEWHEIGHT_OFFSET; //NOTE: the force power checks will stop fencers and grunts from getting up using force jump switch ( pm->ps->legsAnim ) { case BOTH_KNOCKDOWN1: if ( (pm->ps->clientNum&&pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_0) || ((pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer())&&pm->cmd.upmove>0&&pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_0) )//FORCE_LEVEL_1) )FORCE_LEVEL_1) ) { anim = Q_irand( BOTH_FORCE_GETUP_B1, BOTH_FORCE_GETUP_B6 );//NOTE: BOTH_FORCE_GETUP_B5 takes soe steps forward at end forceGetUp = qtrue; } else { anim = BOTH_GETUP1; } break; case BOTH_KNOCKDOWN2: case BOTH_PLAYER_PA_3_FLY: if ( (pm->ps->clientNum&&pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_0) || ((pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer())&&pm->cmd.upmove>0&&pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_0) )//FORCE_LEVEL_1) )FORCE_LEVEL_1) ) { anim = Q_irand( BOTH_FORCE_GETUP_B1, BOTH_FORCE_GETUP_B6 );//NOTE: BOTH_FORCE_GETUP_B5 takes soe steps forward at end forceGetUp = qtrue; } else { anim = BOTH_GETUP2; } break; case BOTH_KNOCKDOWN3: if ( (pm->ps->clientNum&&pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_0) || ((pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer())&&pm->cmd.upmove>0&&pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_0) )//FORCE_LEVEL_1) ) { anim = Q_irand( BOTH_FORCE_GETUP_F1, BOTH_FORCE_GETUP_F2 ); forceGetUp = qtrue; } else { anim = BOTH_GETUP3; } break; case BOTH_KNOCKDOWN4: case BOTH_RELEASED: if ( (pm->ps->clientNum&&pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_0) || ((pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer())&&pm->cmd.upmove>0&&pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_0) )//FORCE_LEVEL_1) )FORCE_LEVEL_1) ) { anim = Q_irand( BOTH_FORCE_GETUP_B1, BOTH_FORCE_GETUP_B6 );//NOTE: BOTH_FORCE_GETUP_B5 takes soe steps forward at end forceGetUp = qtrue; } else { anim = BOTH_GETUP4; } break; case BOTH_KNOCKDOWN5: case BOTH_LK_DL_ST_T_SB_1_L: if ( (pm->ps->clientNum&&pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_0) || ((pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer())&&pm->cmd.upmove>0&&pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_0) )//FORCE_LEVEL_1) )FORCE_LEVEL_1) ) { anim = Q_irand( BOTH_FORCE_GETUP_F1, BOTH_FORCE_GETUP_F2 ); forceGetUp = qtrue; } else { anim = BOTH_GETUP5; } break; } //Com_Printf( "getupanim = %s\n", animTable[anim].name ); if ( forceGetUp ) { if ( pm->gent && pm->gent->client && pm->gent->client->playerTeam == TEAM_ENEMY && pm->gent->NPC && pm->gent->NPC->blockedSpeechDebounceTime < level.time && !Q_irand( 0, 1 ) ) { PM_AddEvent( Q_irand( EV_COMBAT1, EV_COMBAT3 ) ); pm->gent->NPC->blockedSpeechDebounceTime = level.time + 1000; } G_SoundOnEnt( pm->gent, CHAN_BODY, "sound/weapons/force/jump.wav" ); //launch off ground? pm->ps->weaponTime = 300;//just to make sure it's cleared } if ( PM_LockedAnim( pm->ps->torsoAnim ) ) {//need to be able to override this anim pm->ps->torsoAnimTimer = 0; } if ( PM_LockedAnim( pm->ps->legsAnim ) ) {//need to be able to override this anim pm->ps->legsAnimTimer = 0; } PM_SetAnim( pm, SETANIM_BOTH, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD|SETANIM_FLAG_HOLDLESS ); pm->ps->saberMove = pm->ps->saberBounceMove = LS_READY;//don't finish whatever saber anim you may have been in pm->ps->saberBlocked = BLOCKED_NONE; return qtrue; } else { return PM_CrouchGetup( crouchheight ); } } } else { if ( pm->ps->legsAnim == BOTH_LK_DL_ST_T_SB_1_L ) { PM_CmdForRoll( pm->ps, &pm->cmd ); } else { pm->cmd.rightmove = pm->cmd.forwardmove = 0; } } } return qfalse; } void PM_CmdForRoll( playerState_t *ps, usercmd_t *pCmd ) { switch ( ps->legsAnim ) { case BOTH_ROLL_F: pCmd->forwardmove = 127; pCmd->rightmove = 0; break; case BOTH_ROLL_B: pCmd->forwardmove = -127; pCmd->rightmove = 0; break; case BOTH_ROLL_R: pCmd->forwardmove = 0; pCmd->rightmove = 127; break; case BOTH_ROLL_L: pCmd->forwardmove = 0; pCmd->rightmove = -127; break; case BOTH_GETUP_BROLL_R: pCmd->forwardmove = 0; pCmd->rightmove = 48; //NOTE: speed is 400 break; case BOTH_GETUP_FROLL_R: if ( ps->legsAnimTimer <= 250 ) {//end of anim pCmd->forwardmove = pCmd->rightmove = 0; } else { pCmd->forwardmove = 0; pCmd->rightmove = 48; //NOTE: speed is 400 } break; case BOTH_GETUP_BROLL_L: pCmd->forwardmove = 0; pCmd->rightmove = -48; //NOTE: speed is 400 break; case BOTH_GETUP_FROLL_L: if ( ps->legsAnimTimer <= 250 ) {//end of anim pCmd->forwardmove = pCmd->rightmove = 0; } else { pCmd->forwardmove = 0; pCmd->rightmove = -48; //NOTE: speed is 400 } break; case BOTH_GETUP_BROLL_B: if ( ps->torsoAnimTimer <= 250 ) {//end of anim pCmd->forwardmove = pCmd->rightmove = 0; } else if ( PM_AnimLength( g_entities[ps->clientNum].client->clientInfo.animFileIndex, (animNumber_t)ps->legsAnim ) - ps->torsoAnimTimer < 350 ) {//beginning of anim pCmd->forwardmove = pCmd->rightmove = 0; } else { //FIXME: ramp down over length of anim pCmd->forwardmove = -64; pCmd->rightmove = 0; //NOTE: speed is 400 } break; case BOTH_GETUP_FROLL_B: if ( ps->torsoAnimTimer <= 100 ) {//end of anim pCmd->forwardmove = pCmd->rightmove = 0; } else if ( PM_AnimLength( g_entities[ps->clientNum].client->clientInfo.animFileIndex, (animNumber_t)ps->legsAnim ) - ps->torsoAnimTimer < 200 ) {//beginning of anim pCmd->forwardmove = pCmd->rightmove = 0; } else { //FIXME: ramp down over length of anim pCmd->forwardmove = -64; pCmd->rightmove = 0; //NOTE: speed is 400 } break; case BOTH_GETUP_BROLL_F: if ( ps->torsoAnimTimer <= 550 ) {//end of anim pCmd->forwardmove = pCmd->rightmove = 0; } else if ( PM_AnimLength( g_entities[ps->clientNum].client->clientInfo.animFileIndex, (animNumber_t)ps->legsAnim ) - ps->torsoAnimTimer < 150 ) {//beginning of anim pCmd->forwardmove = pCmd->rightmove = 0; } else { pCmd->forwardmove = 64; pCmd->rightmove = 0; //NOTE: speed is 400 } break; case BOTH_GETUP_FROLL_F: if ( ps->torsoAnimTimer <= 100 ) {//end of anim pCmd->forwardmove = pCmd->rightmove = 0; } else { //FIXME: ramp down over length of anim pCmd->forwardmove = 64; pCmd->rightmove = 0; //NOTE: speed is 400 } break; case BOTH_LK_DL_ST_T_SB_1_L: //kicked backwards if ( ps->legsAnimTimer < 3050//at least 10 frames in && ps->legsAnimTimer > 550 )//at least 6 frames from end {//move backwards pCmd->forwardmove = -64; pCmd->rightmove = 0; } else { pCmd->forwardmove = pCmd->rightmove = 0; } break; } pCmd->upmove = 0; } qboolean PM_InRollIgnoreTimer( playerState_t *ps ) { switch ( ps->legsAnim ) { case BOTH_ROLL_F: case BOTH_ROLL_B: case BOTH_ROLL_R: case BOTH_ROLL_L: case BOTH_GETUP_BROLL_B: case BOTH_GETUP_BROLL_F: case BOTH_GETUP_BROLL_L: case BOTH_GETUP_BROLL_R: case BOTH_GETUP_FROLL_B: case BOTH_GETUP_FROLL_F: case BOTH_GETUP_FROLL_L: case BOTH_GETUP_FROLL_R: return qtrue; } return qfalse; } qboolean PM_InRoll( playerState_t *ps ) { if ( ps->legsAnimTimer && PM_InRollIgnoreTimer( ps ) ) { return qtrue; } return qfalse; } qboolean PM_CrouchAnim( int anim ) { switch ( anim ) { case BOTH_SIT1: //# Normal chair sit. case BOTH_SIT2: //# Lotus position. case BOTH_SIT3: //# Sitting in tired position: elbows on knees case BOTH_CROUCH1: //# Transition from standing to crouch case BOTH_CROUCH1IDLE: //# Crouching idle case BOTH_CROUCH1WALK: //# Walking while crouched case BOTH_CROUCH1WALKBACK: //# Walking while crouched case BOTH_CROUCH2TOSTAND1: //# going from crouch2 to stand1 case BOTH_CROUCH3: //# Desann crouching down to Kyle (cin 9) case BOTH_KNEES1: //# Tavion on her knees case BOTH_CROUCHATTACKBACK1://FIXME: not if in middle of anim? case BOTH_ROLL_STAB: //??? case BOTH_STAND_TO_KNEEL: case BOTH_KNEEL_TO_STAND: case BOTH_TURNCROUCH1: case BOTH_CROUCH4: case BOTH_KNEES2: //# Tavion on her knees looking down case BOTH_KNEES2TO1: //# Transition of KNEES2 to KNEES1 return qtrue; break; } return qfalse; } qboolean PM_PainAnim( int anim ) { switch ( anim ) { case BOTH_PAIN1: //# First take pain anim case BOTH_PAIN2: //# Second take pain anim case BOTH_PAIN3: //# Third take pain anim case BOTH_PAIN4: //# Fourth take pain anim case BOTH_PAIN5: //# Fifth take pain anim - from behind case BOTH_PAIN6: //# Sixth take pain anim - from behind case BOTH_PAIN7: //# Seventh take pain anim - from behind case BOTH_PAIN8: //# Eigth take pain anim - from behind case BOTH_PAIN9: //# case BOTH_PAIN10: //# case BOTH_PAIN11: //# case BOTH_PAIN12: //# case BOTH_PAIN13: //# case BOTH_PAIN14: //# case BOTH_PAIN15: //# case BOTH_PAIN16: //# case BOTH_PAIN17: //# case BOTH_PAIN18: //# return qtrue; break; } return qfalse; } qboolean PM_DodgeHoldAnim( int anim ) { switch ( anim ) { case BOTH_DODGE_HOLD_FL: case BOTH_DODGE_HOLD_FR: case BOTH_DODGE_HOLD_BL: case BOTH_DODGE_HOLD_BR: case BOTH_DODGE_HOLD_L: case BOTH_DODGE_HOLD_R: return qtrue; break; } return qfalse; } qboolean PM_DodgeAnim( int anim ) { switch ( anim ) { case BOTH_DODGE_FL: //# lean-dodge forward left case BOTH_DODGE_FR: //# lean-dodge forward right case BOTH_DODGE_BL: //# lean-dodge backwards left case BOTH_DODGE_BR: //# lean-dodge backwards right case BOTH_DODGE_L: //# lean-dodge left case BOTH_DODGE_R: //# lean-dodge right return qtrue; break; default: return PM_DodgeHoldAnim( anim ); break; } //return qfalse; } qboolean PM_ForceJumpingAnim( int anim ) { switch ( anim ) { case BOTH_FORCEJUMP1: //# Jump - wind-up and leave ground case BOTH_FORCEINAIR1: //# In air loop (from jump) case BOTH_FORCELAND1: //# Landing (from in air loop) case BOTH_FORCEJUMPBACK1: //# Jump backwards - wind-up and leave ground case BOTH_FORCEINAIRBACK1: //# In air loop (from jump back) case BOTH_FORCELANDBACK1: //# Landing backwards(from in air loop) case BOTH_FORCEJUMPLEFT1: //# Jump left - wind-up and leave ground case BOTH_FORCEINAIRLEFT1: //# In air loop (from jump left) case BOTH_FORCELANDLEFT1: //# Landing left(from in air loop) case BOTH_FORCEJUMPRIGHT1: //# Jump right - wind-up and leave ground case BOTH_FORCEINAIRRIGHT1: //# In air loop (from jump right) case BOTH_FORCELANDRIGHT1: //# Landing right(from in air loop) return qtrue; break; } return qfalse; } qboolean PM_JumpingAnim( int anim ) { switch ( anim ) { case BOTH_JUMP1: //# Jump - wind-up and leave ground case BOTH_INAIR1: //# In air loop (from jump) case BOTH_LAND1: //# Landing (from in air loop) case BOTH_LAND2: //# Landing Hard (from a great height) case BOTH_JUMPBACK1: //# Jump backwards - wind-up and leave ground case BOTH_INAIRBACK1: //# In air loop (from jump back) case BOTH_LANDBACK1: //# Landing backwards(from in air loop) case BOTH_JUMPLEFT1: //# Jump left - wind-up and leave ground case BOTH_INAIRLEFT1: //# In air loop (from jump left) case BOTH_LANDLEFT1: //# Landing left(from in air loop) case BOTH_JUMPRIGHT1: //# Jump right - wind-up and leave ground case BOTH_INAIRRIGHT1: //# In air loop (from jump right) case BOTH_LANDRIGHT1: //# Landing right(from in air loop) return qtrue; break; default: if ( PM_InAirKickingAnim( anim ) ) { return qtrue; } return PM_ForceJumpingAnim( anim ); break; } //return qfalse; } qboolean PM_LandingAnim( int anim ) { switch ( anim ) { case BOTH_LAND1: //# Landing (from in air loop) case BOTH_LAND2: //# Landing Hard (from a great height) case BOTH_LANDBACK1: //# Landing backwards(from in air loop) case BOTH_LANDLEFT1: //# Landing left(from in air loop) case BOTH_LANDRIGHT1: //# Landing right(from in air loop) case BOTH_FORCELAND1: //# Landing (from in air loop) case BOTH_FORCELANDBACK1: //# Landing backwards(from in air loop) case BOTH_FORCELANDLEFT1: //# Landing left(from in air loop) case BOTH_FORCELANDRIGHT1: //# Landing right(from in air loop) return qtrue; break; } return qfalse; } qboolean PM_FlippingAnim( int anim ) { switch ( anim ) { case BOTH_FLIP_F: //# Flip forward case BOTH_FLIP_B: //# Flip backwards case BOTH_FLIP_L: //# Flip left case BOTH_FLIP_R: //# Flip right case BOTH_ALORA_FLIP_1: case BOTH_ALORA_FLIP_2: case BOTH_ALORA_FLIP_3: case BOTH_WALL_RUN_RIGHT_FLIP: case BOTH_WALL_RUN_LEFT_FLIP: case BOTH_WALL_FLIP_RIGHT: case BOTH_WALL_FLIP_LEFT: case BOTH_FLIP_BACK1: case BOTH_FLIP_BACK2: case BOTH_FLIP_BACK3: case BOTH_WALL_FLIP_BACK1: case BOTH_ALORA_FLIP_B: //Not really flips, but... case BOTH_WALL_RUN_RIGHT: case BOTH_WALL_RUN_LEFT: case BOTH_WALL_RUN_RIGHT_STOP: case BOTH_WALL_RUN_LEFT_STOP: case BOTH_BUTTERFLY_LEFT: case BOTH_BUTTERFLY_RIGHT: case BOTH_BUTTERFLY_FL1: case BOTH_BUTTERFLY_FR1: // case BOTH_ARIAL_LEFT: case BOTH_ARIAL_RIGHT: case BOTH_ARIAL_F1: case BOTH_CARTWHEEL_LEFT: case BOTH_CARTWHEEL_RIGHT: case BOTH_JUMPFLIPSLASHDOWN1: case BOTH_JUMPFLIPSTABDOWN: case BOTH_JUMPATTACK6: case BOTH_JUMPATTACK7: //JKA case BOTH_FORCEWALLRUNFLIP_END: case BOTH_FORCEWALLRUNFLIP_ALT: case BOTH_FLIP_ATTACK7: case BOTH_A7_SOULCAL: return qtrue; break; } return qfalse; } qboolean PM_WalkingAnim( int anim ) { switch ( anim ) { case BOTH_WALK1: //# Normal walk case BOTH_WALK2: //# Normal walk with saber case BOTH_WALK_STAFF: //# Normal walk with staff case BOTH_WALK_DUAL: //# Normal walk with staff case BOTH_WALK5: //# Tavion taunting Kyle (cin 22) case BOTH_WALK6: //# Slow walk for Luke (cin 12) case BOTH_WALK7: //# Fast walk case BOTH_WALKBACK1: //# Walk1 backwards case BOTH_WALKBACK2: //# Walk2 backwards case BOTH_WALKBACK_STAFF: //# Walk backwards with staff case BOTH_WALKBACK_DUAL: //# Walk backwards with dual return qtrue; break; } return qfalse; } qboolean PM_RunningAnim( int anim ) { switch ( anim ) { case BOTH_RUN1: case BOTH_RUN2: case BOTH_RUN4: case BOTH_RUN_STAFF: case BOTH_RUN_DUAL: case BOTH_RUNBACK1: case BOTH_RUNBACK2: case BOTH_RUNBACK_STAFF: case BOTH_RUN1START: //# Start into full run1 case BOTH_RUN1STOP: //# Stop from full run1 case BOTH_RUNSTRAFE_LEFT1: //# Sidestep left: should loop case BOTH_RUNSTRAFE_RIGHT1: //# Sidestep right: should loop return qtrue; break; } return qfalse; } qboolean PM_RollingAnim( int anim ) { switch ( anim ) { case BOTH_ROLL_F: //# Roll forward case BOTH_ROLL_B: //# Roll backward case BOTH_ROLL_L: //# Roll left case BOTH_ROLL_R: //# Roll right case BOTH_GETUP_BROLL_B: case BOTH_GETUP_BROLL_F: case BOTH_GETUP_BROLL_L: case BOTH_GETUP_BROLL_R: case BOTH_GETUP_FROLL_B: case BOTH_GETUP_FROLL_F: case BOTH_GETUP_FROLL_L: case BOTH_GETUP_FROLL_R: return qtrue; break; } return qfalse; } qboolean PM_SwimmingAnim( int anim ) { switch ( anim ) { case BOTH_SWIM_IDLE1: //# Swimming Idle 1 case BOTH_SWIMFORWARD: //# Swim forward loop case BOTH_SWIMBACKWARD: //# Swim backward loop return qtrue; break; } return qfalse; } qboolean PM_LeapingSaberAnim( int anim ) { switch ( anim ) { //level 7 case BOTH_T7_BR_TL: case BOTH_T7__L_BR: case BOTH_T7__L__R: case BOTH_T7_BL_BR: case BOTH_T7_BL__R: case BOTH_T7_BL_TR: return qtrue; break; } return qfalse; } qboolean PM_SpinningSaberAnim( int anim ) { switch ( anim ) { //level 1 - FIXME: level 1 will have *no* spins case BOTH_T1_BR_BL: case BOTH_T1__R__L: case BOTH_T1__R_BL: case BOTH_T1_TR_BL: case BOTH_T1_BR_TL: case BOTH_T1_BR__L: case BOTH_T1_TL_BR: case BOTH_T1__L_BR: case BOTH_T1__L__R: case BOTH_T1_BL_BR: case BOTH_T1_BL__R: case BOTH_T1_BL_TR: //level 2 case BOTH_T2_BR__L: case BOTH_T2_BR_BL: case BOTH_T2__R_BL: case BOTH_T2__L_BR: case BOTH_T2_BL_BR: case BOTH_T2_BL__R: //level 3 case BOTH_T3_BR__L: case BOTH_T3_BR_BL: case BOTH_T3__R_BL: case BOTH_T3__L_BR: case BOTH_T3_BL_BR: case BOTH_T3_BL__R: //level 4 case BOTH_T4_BR__L: case BOTH_T4_BR_BL: case BOTH_T4__R_BL: case BOTH_T4__L_BR: case BOTH_T4_BL_BR: case BOTH_T4_BL__R: //level 5 case BOTH_T5_BR_BL: case BOTH_T5__R__L: case BOTH_T5__R_BL: case BOTH_T5_TR_BL: case BOTH_T5_BR_TL: case BOTH_T5_BR__L: case BOTH_T5_TL_BR: case BOTH_T5__L_BR: case BOTH_T5__L__R: case BOTH_T5_BL_BR: case BOTH_T5_BL__R: case BOTH_T5_BL_TR: //level 6 case BOTH_T6_BR_TL: case BOTH_T6__R_TL: case BOTH_T6__R__L: case BOTH_T6__R_BL: case BOTH_T6_TR_TL: case BOTH_T6_TR__L: case BOTH_T6_TR_BL: case BOTH_T6_T__TL: case BOTH_T6_T__BL: case BOTH_T6_TL_BR: case BOTH_T6__L_BR: case BOTH_T6__L__R: case BOTH_T6_TL__R: case BOTH_T6_TL_TR: case BOTH_T6__L_TR: case BOTH_T6__L_T_: case BOTH_T6_BL_T_: case BOTH_T6_BR__L: case BOTH_T6_BR_BL: case BOTH_T6_BL_BR: case BOTH_T6_BL__R: case BOTH_T6_BL_TR: //level 7 case BOTH_T7_BR_TL: case BOTH_T7_BR__L: case BOTH_T7_BR_BL: case BOTH_T7__R__L: case BOTH_T7__R_BL: case BOTH_T7_TR__L: case BOTH_T7_T___R: case BOTH_T7_TL_BR: case BOTH_T7__L_BR: case BOTH_T7__L__R: case BOTH_T7_BL_BR: case BOTH_T7_BL__R: case BOTH_T7_BL_TR: case BOTH_T7_TL_TR: case BOTH_T7_T__BR: case BOTH_T7__L_TR: case BOTH_V7_BL_S7: //special //case BOTH_A2_STABBACK1: case BOTH_ATTACK_BACK: case BOTH_CROUCHATTACKBACK1: case BOTH_BUTTERFLY_LEFT: case BOTH_BUTTERFLY_RIGHT: case BOTH_FJSS_TR_BL: case BOTH_FJSS_TL_BR: case BOTH_JUMPFLIPSLASHDOWN1: case BOTH_JUMPFLIPSTABDOWN: return qtrue; break; } return qfalse; } qboolean PM_SpinningAnim( int anim ) { /* switch ( anim ) { //FIXME: list any other spinning anims default: break; } */ return PM_SpinningSaberAnim( anim ); } void PM_ResetAnkleAngles( void ) { if ( !pm->gent || !pm->gent->client || pm->gent->client->NPC_class != CLASS_ATST ) { return; } if ( pm->gent->footLBone != -1 ) { gi.G2API_SetBoneAnglesIndex( &pm->gent->ghoul2[0], pm->gent->footLBone, vec3_origin, BONE_ANGLES_POSTMULT, POSITIVE_Z, NEGATIVE_Y, NEGATIVE_X, NULL, 0, 0 ); } if ( pm->gent->footRBone != -1 ) { gi.G2API_SetBoneAnglesIndex( &pm->gent->ghoul2[0], pm->gent->footRBone, vec3_origin, BONE_ANGLES_POSTMULT, POSITIVE_Z, NEGATIVE_Y, NEGATIVE_X, NULL, 0, 0 ); } } void PM_AnglesForSlope( const float yaw, const vec3_t slope, vec3_t angles ) { vec3_t nvf, ovf, ovr, new_angles; float pitch, mod, dot; VectorSet( angles, 0, yaw, 0 ); AngleVectors( angles, ovf, ovr, NULL ); vectoangles( slope, new_angles ); pitch = new_angles[PITCH] + 90; new_angles[ROLL] = new_angles[PITCH] = 0; AngleVectors( new_angles, nvf, NULL, NULL ); mod = DotProduct( nvf, ovr ); if ( mod < 0 ) mod = -1; else mod = 1; dot = DotProduct( nvf, ovf ); angles[YAW] = 0; angles[PITCH] = dot * pitch; angles[ROLL] = ((1-Q_fabs(dot)) * pitch * mod); } void PM_FootSlopeTrace( float *pDiff, float *pInterval ) { vec3_t footLOrg, footROrg, footLBot, footRBot; trace_t trace; float diff, interval; if ( pm->gent->client->NPC_class == CLASS_ATST ) { interval = 10; } else { interval = 4;//? } if ( pm->gent->footLBolt == -1 || pm->gent->footRBolt == -1 ) { if ( pDiff != NULL ) { *pDiff = 0; } if ( pInterval != NULL ) { *pInterval = interval; } return; } #if 1 for ( int i = 0; i < 3; i++ ) { if ( Q_isnan( pm->gent->client->renderInfo.footLPoint[i] ) || Q_isnan( pm->gent->client->renderInfo.footRPoint[i] ) ) { if ( pDiff != NULL ) { *pDiff = 0; } if ( pInterval != NULL ) { *pInterval = interval; } return; } } #else //FIXME: these really should have been gotten on the cgame, but I guess sometimes they're not and we end up with qnan numbers! mdxaBone_t boltMatrix; vec3_t G2Angles = {0, pm->gent->client->ps.legsYaw, 0}; //get the feet gi.G2API_GetBoltMatrix( pm->gent->ghoul2, pm->gent->playerModel, pm->gent->footLBolt, &boltMatrix, G2Angles, pm->ps->origin, (cg.time?cg.time:level.time), NULL, pm->gent->s.modelScale ); gi.G2API_GiveMeVectorFromMatrix( boltMatrix, ORIGIN, pm->gent->client->renderInfo.footLPoint ); gi.G2API_GetBoltMatrix( pm->gent->ghoul2, pm->gent->playerModel, pm->gent->footRBolt, &boltMatrix, G2Angles, pm->ps->origin, (cg.time?cg.time:level.time), NULL, pm->gent->s.modelScale ); gi.G2API_GiveMeVectorFromMatrix( boltMatrix, ORIGIN, pm->gent->client->renderInfo.footRPoint ); #endif //NOTE: on AT-STs, rotating the foot moves this point, so it will wiggle... // we have to do this extra work (more G2 transforms) to stop the wiggle... is it worth it? /* if ( pm->gent->client->NPC_class == CLASS_ATST ) { mdxaBone_t boltMatrix; vec3_t G2Angles = {0, pm->gent->client->ps.legsYaw, 0}; //get the feet gi.G2API_SetBoneAnglesIndex( &pm->gent->ghoul2[0], pm->gent->footLBone, vec3_origin, BONE_ANGLES_POSTMULT, POSITIVE_Z, NEGATIVE_Y, NEGATIVE_X, NULL ); gi.G2API_GetBoltMatrix( pm->gent->ghoul2, pm->gent->playerModel, pm->gent->footLBolt, &boltMatrix, G2Angles, pm->ps->origin, (cg.time?cg.time:level.time), NULL, pm->gent->s.modelScale ); gi.G2API_GiveMeVectorFromMatrix( boltMatrix, ORIGIN, pm->gent->client->renderInfo.footLPoint ); gi.G2API_SetBoneAnglesIndex( &pm->gent->ghoul2[0], pm->gent->footRBone, vec3_origin, BONE_ANGLES_POSTMULT, POSITIVE_Z, NEGATIVE_Y, NEGATIVE_X, NULL ); gi.G2API_GetBoltMatrix( pm->gent->ghoul2, pm->gent->playerModel, pm->gent->footRBolt, &boltMatrix, G2Angles, pm->ps->origin, (cg.time?cg.time:level.time), NULL, pm->gent->s.modelScale ); gi.G2API_GiveMeVectorFromMatrix( boltMatrix, ORIGIN, pm->gent->client->renderInfo.footRPoint ); } */ //get these on the cgame and store it, save ourselves a ghoul2 construct skel call VectorCopy( pm->gent->client->renderInfo.footLPoint, footLOrg ); VectorCopy( pm->gent->client->renderInfo.footRPoint, footROrg ); //step 2: adjust foot tag z height to bottom of bbox+1 footLOrg[2] = pm->gent->currentOrigin[2] + pm->gent->mins[2] + 1; footROrg[2] = pm->gent->currentOrigin[2] + pm->gent->mins[2] + 1; VectorSet( footLBot, footLOrg[0], footLOrg[1], footLOrg[2] - interval*10 ); VectorSet( footRBot, footROrg[0], footROrg[1], footROrg[2] - interval*10 ); //step 3: trace down from each, find difference vec3_t footMins, footMaxs; vec3_t footLSlope, footRSlope; if ( pm->gent->client->NPC_class == CLASS_ATST ) { VectorSet( footMins, -16, -16, 0 ); VectorSet( footMaxs, 16, 16, 1 ); } else { VectorSet( footMins, -3, -3, 0 ); VectorSet( footMaxs, 3, 3, 1 ); } pm->trace( &trace, footLOrg, footMins, footMaxs, footLBot, pm->ps->clientNum, pm->tracemask, (EG2_Collision)0, 0 ); VectorCopy( trace.endpos, footLBot ); VectorCopy( trace.plane.normal, footLSlope ); pm->trace( &trace, footROrg, footMins, footMaxs, footRBot, pm->ps->clientNum, pm->tracemask, (EG2_Collision)0, 0 ); VectorCopy( trace.endpos, footRBot ); VectorCopy( trace.plane.normal, footRSlope ); diff = footLBot[2] - footRBot[2]; //optional step: for atst, tilt the footpads to match the slopes under it... if ( pm->gent->client->NPC_class == CLASS_ATST ) { vec3_t footAngles; if ( !VectorCompare( footLSlope, vec3_origin ) ) {//rotate the ATST's left foot pad to match the slope PM_AnglesForSlope( pm->gent->client->renderInfo.legsYaw, footLSlope, footAngles ); //Hmm... lerp this? gi.G2API_SetBoneAnglesIndex( &pm->gent->ghoul2[0], pm->gent->footLBone, footAngles, BONE_ANGLES_POSTMULT, POSITIVE_Z, NEGATIVE_Y, NEGATIVE_X, NULL, 0, 0 ); } if ( !VectorCompare( footRSlope, vec3_origin ) ) {//rotate the ATST's right foot pad to match the slope PM_AnglesForSlope( pm->gent->client->renderInfo.legsYaw, footRSlope, footAngles ); //Hmm... lerp this? gi.G2API_SetBoneAnglesIndex( &pm->gent->ghoul2[0], pm->gent->footRBone, footAngles, BONE_ANGLES_POSTMULT, POSITIVE_Z, NEGATIVE_Y, NEGATIVE_X, NULL, 0, 0 ); } } if ( pDiff != NULL ) { *pDiff = diff; } if ( pInterval != NULL ) { *pInterval = interval; } } qboolean PM_InSlopeAnim( int anim ) { switch ( anim ) { case LEGS_LEFTUP1: //# On a slope with left foot 4 higher than right case LEGS_LEFTUP2: //# On a slope with left foot 8 higher than right case LEGS_LEFTUP3: //# On a slope with left foot 12 higher than right case LEGS_LEFTUP4: //# On a slope with left foot 16 higher than right case LEGS_LEFTUP5: //# On a slope with left foot 20 higher than right case LEGS_RIGHTUP1: //# On a slope with RIGHT foot 4 higher than left case LEGS_RIGHTUP2: //# On a slope with RIGHT foot 8 higher than left case LEGS_RIGHTUP3: //# On a slope with RIGHT foot 12 higher than left case LEGS_RIGHTUP4: //# On a slope with RIGHT foot 16 higher than left case LEGS_RIGHTUP5: //# On a slope with RIGHT foot 20 higher than left case LEGS_S1_LUP1: case LEGS_S1_LUP2: case LEGS_S1_LUP3: case LEGS_S1_LUP4: case LEGS_S1_LUP5: case LEGS_S1_RUP1: case LEGS_S1_RUP2: case LEGS_S1_RUP3: case LEGS_S1_RUP4: case LEGS_S1_RUP5: case LEGS_S3_LUP1: case LEGS_S3_LUP2: case LEGS_S3_LUP3: case LEGS_S3_LUP4: case LEGS_S3_LUP5: case LEGS_S3_RUP1: case LEGS_S3_RUP2: case LEGS_S3_RUP3: case LEGS_S3_RUP4: case LEGS_S3_RUP5: case LEGS_S4_LUP1: case LEGS_S4_LUP2: case LEGS_S4_LUP3: case LEGS_S4_LUP4: case LEGS_S4_LUP5: case LEGS_S4_RUP1: case LEGS_S4_RUP2: case LEGS_S4_RUP3: case LEGS_S4_RUP4: case LEGS_S4_RUP5: case LEGS_S5_LUP1: case LEGS_S5_LUP2: case LEGS_S5_LUP3: case LEGS_S5_LUP4: case LEGS_S5_LUP5: case LEGS_S5_RUP1: case LEGS_S5_RUP2: case LEGS_S5_RUP3: case LEGS_S5_RUP4: case LEGS_S5_RUP5: case LEGS_S6_LUP1: case LEGS_S6_LUP2: case LEGS_S6_LUP3: case LEGS_S6_LUP4: case LEGS_S6_LUP5: case LEGS_S6_RUP1: case LEGS_S6_RUP2: case LEGS_S6_RUP3: case LEGS_S6_RUP4: case LEGS_S6_RUP5: case LEGS_S7_LUP1: case LEGS_S7_LUP2: case LEGS_S7_LUP3: case LEGS_S7_LUP4: case LEGS_S7_LUP5: case LEGS_S7_RUP1: case LEGS_S7_RUP2: case LEGS_S7_RUP3: case LEGS_S7_RUP4: case LEGS_S7_RUP5: return qtrue; break; } return qfalse; } qboolean PM_SaberStanceAnim( int anim ) { switch ( anim ) { case BOTH_STAND1://not really a saberstance anim, actually... "saber off" stance case BOTH_STAND2://single-saber, medium style case BOTH_SABERFAST_STANCE://single-saber, fast style case BOTH_SABERSLOW_STANCE://single-saber, strong style case BOTH_SABERSTAFF_STANCE://saber staff style case BOTH_SABERDUAL_STANCE://dual saber style return qtrue; break; } return qfalse; } qboolean PM_SaberDrawPutawayAnim( int anim ) { switch ( anim ) { case BOTH_STAND1TO2: case BOTH_STAND2TO1: case BOTH_S1_S7: case BOTH_S7_S1: case BOTH_S1_S6: case BOTH_S6_S1: return qtrue; break; } return qfalse; } #define SLOPE_RECALC_INT 100 extern qboolean G_StandardHumanoid( gentity_t *self ); qboolean PM_AdjustStandAnimForSlope( void ) { if ( !pm->gent || !pm->gent->client ) { return qfalse; } if ( pm->gent->client->NPC_class != CLASS_ATST && (!pm->gent||!G_StandardHumanoid( pm->gent )) ) {//only ATST and player does this return qfalse; } if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && (!cg.renderingThirdPerson || cg.zoomMode) ) {//first person doesn't do this return qfalse; } if ( pm->gent->footLBolt == -1 || pm->gent->footRBolt == -1 ) {//need these bolts! return qfalse; } //step 1: find the 2 foot tags float diff; float interval; PM_FootSlopeTrace( &diff, &interval ); //step 4: based on difference, choose one of the left/right slope-match intervals int destAnim; if ( diff >= interval*5 ) { destAnim = LEGS_LEFTUP5; } else if ( diff >= interval*4 ) { destAnim = LEGS_LEFTUP4; } else if ( diff >= interval*3 ) { destAnim = LEGS_LEFTUP3; } else if ( diff >= interval*2 ) { destAnim = LEGS_LEFTUP2; } else if ( diff >= interval ) { destAnim = LEGS_LEFTUP1; } else if ( diff <= interval*-5 ) { destAnim = LEGS_RIGHTUP5; } else if ( diff <= interval*-4 ) { destAnim = LEGS_RIGHTUP4; } else if ( diff <= interval*-3 ) { destAnim = LEGS_RIGHTUP3; } else if ( diff <= interval*-2 ) { destAnim = LEGS_RIGHTUP2; } else if ( diff <= interval*-1 ) { destAnim = LEGS_RIGHTUP1; } else { return qfalse; } int legsAnim = pm->ps->legsAnim; if ( pm->gent->client->NPC_class != CLASS_ATST ) { //adjust for current legs anim switch ( legsAnim ) { case BOTH_STAND1: case LEGS_S1_LUP1: case LEGS_S1_LUP2: case LEGS_S1_LUP3: case LEGS_S1_LUP4: case LEGS_S1_LUP5: case LEGS_S1_RUP1: case LEGS_S1_RUP2: case LEGS_S1_RUP3: case LEGS_S1_RUP4: case LEGS_S1_RUP5: destAnim = LEGS_S1_LUP1 + (destAnim-LEGS_LEFTUP1); break; case BOTH_STAND2: case BOTH_SABERFAST_STANCE: case BOTH_SABERSLOW_STANCE: case BOTH_CROUCH1IDLE: case BOTH_CROUCH1: case LEGS_LEFTUP1: //# On a slope with left foot 4 higher than right case LEGS_LEFTUP2: //# On a slope with left foot 8 higher than right case LEGS_LEFTUP3: //# On a slope with left foot 12 higher than right case LEGS_LEFTUP4: //# On a slope with left foot 16 higher than right case LEGS_LEFTUP5: //# On a slope with left foot 20 higher than right case LEGS_RIGHTUP1: //# On a slope with RIGHT foot 4 higher than left case LEGS_RIGHTUP2: //# On a slope with RIGHT foot 8 higher than left case LEGS_RIGHTUP3: //# On a slope with RIGHT foot 12 higher than left case LEGS_RIGHTUP4: //# On a slope with RIGHT foot 16 higher than left case LEGS_RIGHTUP5: //# On a slope with RIGHT foot 20 higher than left //fine break; case BOTH_STAND3: case LEGS_S3_LUP1: case LEGS_S3_LUP2: case LEGS_S3_LUP3: case LEGS_S3_LUP4: case LEGS_S3_LUP5: case LEGS_S3_RUP1: case LEGS_S3_RUP2: case LEGS_S3_RUP3: case LEGS_S3_RUP4: case LEGS_S3_RUP5: destAnim = LEGS_S3_LUP1 + (destAnim-LEGS_LEFTUP1); break; case BOTH_STAND4: case LEGS_S4_LUP1: case LEGS_S4_LUP2: case LEGS_S4_LUP3: case LEGS_S4_LUP4: case LEGS_S4_LUP5: case LEGS_S4_RUP1: case LEGS_S4_RUP2: case LEGS_S4_RUP3: case LEGS_S4_RUP4: case LEGS_S4_RUP5: destAnim = LEGS_S4_LUP1 + (destAnim-LEGS_LEFTUP1); break; case BOTH_STAND5: case LEGS_S5_LUP1: case LEGS_S5_LUP2: case LEGS_S5_LUP3: case LEGS_S5_LUP4: case LEGS_S5_LUP5: case LEGS_S5_RUP1: case LEGS_S5_RUP2: case LEGS_S5_RUP3: case LEGS_S5_RUP4: case LEGS_S5_RUP5: destAnim = LEGS_S5_LUP1 + (destAnim-LEGS_LEFTUP1); break; case BOTH_SABERDUAL_STANCE: case LEGS_S6_LUP1: case LEGS_S6_LUP2: case LEGS_S6_LUP3: case LEGS_S6_LUP4: case LEGS_S6_LUP5: case LEGS_S6_RUP1: case LEGS_S6_RUP2: case LEGS_S6_RUP3: case LEGS_S6_RUP4: case LEGS_S6_RUP5: destAnim = LEGS_S6_LUP1 + (destAnim-LEGS_LEFTUP1); break; case BOTH_SABERSTAFF_STANCE: case LEGS_S7_LUP1: case LEGS_S7_LUP2: case LEGS_S7_LUP3: case LEGS_S7_LUP4: case LEGS_S7_LUP5: case LEGS_S7_RUP1: case LEGS_S7_RUP2: case LEGS_S7_RUP3: case LEGS_S7_RUP4: case LEGS_S7_RUP5: destAnim = LEGS_S7_LUP1 + (destAnim-LEGS_LEFTUP1); break; case BOTH_STAND6: default: return qfalse; break; } } //step 5: based on the chosen interval and the current legsAnim, pick the correct anim //step 6: increment/decrement to the dest anim, not instant if ( (legsAnim >= LEGS_LEFTUP1 && legsAnim <= LEGS_LEFTUP5) || (legsAnim >= LEGS_S1_LUP1 && legsAnim <= LEGS_S1_LUP5) || (legsAnim >= LEGS_S3_LUP1 && legsAnim <= LEGS_S3_LUP5) || (legsAnim >= LEGS_S4_LUP1 && legsAnim <= LEGS_S4_LUP5) || (legsAnim >= LEGS_S5_LUP1 && legsAnim <= LEGS_S5_LUP5) || (legsAnim >= LEGS_S6_LUP1 && legsAnim <= LEGS_S6_LUP5) || (legsAnim >= LEGS_S7_LUP1 && legsAnim <= LEGS_S7_LUP5) ) {//already in left-side up if ( destAnim > legsAnim && pm->gent->client->slopeRecalcTime < level.time ) { legsAnim++; pm->gent->client->slopeRecalcTime = level.time + SLOPE_RECALC_INT; } else if ( destAnim < legsAnim && pm->gent->client->slopeRecalcTime < level.time ) { legsAnim--; pm->gent->client->slopeRecalcTime = level.time + SLOPE_RECALC_INT; } else { destAnim = legsAnim; } } else if ( (legsAnim >= LEGS_RIGHTUP1 && legsAnim <= LEGS_RIGHTUP5) || (legsAnim >= LEGS_S1_RUP1 && legsAnim <= LEGS_S1_RUP5) || (legsAnim >= LEGS_S3_RUP1 && legsAnim <= LEGS_S3_RUP5) || (legsAnim >= LEGS_S4_RUP1 && legsAnim <= LEGS_S4_RUP5) || (legsAnim >= LEGS_S5_RUP1 && legsAnim <= LEGS_S5_RUP5) || (legsAnim >= LEGS_S6_RUP1 && legsAnim <= LEGS_S6_RUP5) || (legsAnim >= LEGS_S7_RUP1 && legsAnim <= LEGS_S7_RUP5) ) {//already in right-side up if ( destAnim > legsAnim && pm->gent->client->slopeRecalcTime < level.time ) { legsAnim++; pm->gent->client->slopeRecalcTime = level.time + SLOPE_RECALC_INT; } else if ( destAnim < legsAnim && pm->gent->client->slopeRecalcTime < level.time ) { legsAnim--; pm->gent->client->slopeRecalcTime = level.time + SLOPE_RECALC_INT; } else { destAnim = legsAnim; } } else {//in a stand of some sort? if ( pm->gent->client->NPC_class == CLASS_ATST ) { if ( legsAnim == BOTH_STAND1 || legsAnim == BOTH_STAND2 || legsAnim == BOTH_CROUCH1IDLE ) { if ( destAnim >= LEGS_LEFTUP1 && destAnim <= LEGS_LEFTUP5 ) {//going into left side up destAnim = LEGS_LEFTUP1; pm->gent->client->slopeRecalcTime = level.time + SLOPE_RECALC_INT; } else if ( destAnim >= LEGS_RIGHTUP1 && destAnim <= LEGS_RIGHTUP5 ) {//going into right side up destAnim = LEGS_RIGHTUP1; pm->gent->client->slopeRecalcTime = level.time + SLOPE_RECALC_INT; } else {//will never get here return qfalse; } } } else { switch ( legsAnim ) { case BOTH_STAND1: if ( destAnim >= LEGS_S1_LUP1 && destAnim <= LEGS_S1_LUP5 ) {//going into left side up destAnim = LEGS_S1_LUP1; pm->gent->client->slopeRecalcTime = level.time + SLOPE_RECALC_INT; } else if ( destAnim >= LEGS_S1_RUP1 && destAnim <= LEGS_S1_RUP5 ) {//going into right side up destAnim = LEGS_S1_RUP1; pm->gent->client->slopeRecalcTime = level.time + SLOPE_RECALC_INT; } else {//will never get here return qfalse; } break; case BOTH_STAND2: case BOTH_SABERFAST_STANCE: case BOTH_SABERSLOW_STANCE: case BOTH_CROUCH1IDLE: if ( destAnim >= LEGS_LEFTUP1 && destAnim <= LEGS_LEFTUP5 ) {//going into left side up destAnim = LEGS_LEFTUP1; pm->gent->client->slopeRecalcTime = level.time + SLOPE_RECALC_INT; } else if ( destAnim >= LEGS_RIGHTUP1 && destAnim <= LEGS_RIGHTUP5 ) {//going into right side up destAnim = LEGS_RIGHTUP1; pm->gent->client->slopeRecalcTime = level.time + SLOPE_RECALC_INT; } else {//will never get here return qfalse; } break; case BOTH_STAND3: if ( destAnim >= LEGS_S3_LUP1 && destAnim <= LEGS_S3_LUP5 ) {//going into left side up destAnim = LEGS_S3_LUP1; pm->gent->client->slopeRecalcTime = level.time + SLOPE_RECALC_INT; } else if ( destAnim >= LEGS_S3_RUP1 && destAnim <= LEGS_S3_RUP5 ) {//going into right side up destAnim = LEGS_S3_RUP1; pm->gent->client->slopeRecalcTime = level.time + SLOPE_RECALC_INT; } else {//will never get here return qfalse; } break; case BOTH_STAND4: if ( destAnim >= LEGS_S4_LUP1 && destAnim <= LEGS_S4_LUP5 ) {//going into left side up destAnim = LEGS_S4_LUP1; pm->gent->client->slopeRecalcTime = level.time + SLOPE_RECALC_INT; } else if ( destAnim >= LEGS_S4_RUP1 && destAnim <= LEGS_S4_RUP5 ) {//going into right side up destAnim = LEGS_S4_RUP1; pm->gent->client->slopeRecalcTime = level.time + SLOPE_RECALC_INT; } else {//will never get here return qfalse; } break; case BOTH_STAND5: if ( destAnim >= LEGS_S5_LUP1 && destAnim <= LEGS_S5_LUP5 ) {//going into left side up destAnim = LEGS_S5_LUP1; pm->gent->client->slopeRecalcTime = level.time + SLOPE_RECALC_INT; } else if ( destAnim >= LEGS_S5_RUP1 && destAnim <= LEGS_S5_RUP5 ) {//going into right side up destAnim = LEGS_S5_RUP1; pm->gent->client->slopeRecalcTime = level.time + SLOPE_RECALC_INT; } else {//will never get here return qfalse; } break; case BOTH_SABERDUAL_STANCE: if ( destAnim >= LEGS_S6_LUP1 && destAnim <= LEGS_S6_LUP5 ) {//going into left side up destAnim = LEGS_S6_LUP1; pm->gent->client->slopeRecalcTime = level.time + SLOPE_RECALC_INT; } else if ( destAnim >= LEGS_S6_RUP1 && destAnim <= LEGS_S6_RUP5 ) {//going into right side up destAnim = LEGS_S6_RUP1; pm->gent->client->slopeRecalcTime = level.time + SLOPE_RECALC_INT; } else {//will never get here return qfalse; } break; case BOTH_SABERSTAFF_STANCE: if ( destAnim >= LEGS_S7_LUP1 && destAnim <= LEGS_S7_LUP5 ) {//going into left side up destAnim = LEGS_S7_LUP1; pm->gent->client->slopeRecalcTime = level.time + SLOPE_RECALC_INT; } else if ( destAnim >= LEGS_S7_RUP1 && destAnim <= LEGS_S7_RUP5 ) {//going into right side up destAnim = LEGS_S7_RUP1; pm->gent->client->slopeRecalcTime = level.time + SLOPE_RECALC_INT; } else {//will never get here return qfalse; } break; case BOTH_STAND6: default: return qfalse; break; } } } //step 7: set the anim PM_SetAnim( pm, SETANIM_LEGS, destAnim, SETANIM_FLAG_NORMAL ); return qtrue; } void PM_JetPackAnim( void ) { if ( !PM_ForceJumpingAnim( pm->ps->legsAnim ) )//haven't started forcejump yet { vec3_t facingFwd, facingRight, facingAngles = {0, pm->ps->viewangles[YAW], 0}; int anim = BOTH_FORCEJUMP1; AngleVectors( facingAngles, facingFwd, facingRight, NULL ); float dotR = DotProduct( facingRight, pm->ps->velocity ); float dotF = DotProduct( facingFwd, pm->ps->velocity ); if ( fabs(dotR) > fabs(dotF) * 1.5 ) { if ( dotR > 150 ) { anim = BOTH_FORCEJUMPRIGHT1; } else if ( dotR < -150 ) { anim = BOTH_FORCEJUMPLEFT1; } } else { if ( dotF > 150 ) { anim = BOTH_FORCEJUMP1; } else if ( dotF < -150 ) { anim = BOTH_FORCEJUMPBACK1; } } int parts = SETANIM_BOTH; if ( pm->ps->weaponTime ) {//FIXME: really only care if we're in a saber attack anim... parts = SETANIM_LEGS; } PM_SetAnim( pm, parts, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); } /* else { if ( !pm->ps->legsAnimTimer ) {//not in the middle of a legsAnim int anim = pm->ps->legsAnim; int newAnim = -1; switch ( anim ) { case BOTH_FORCEJUMP1: newAnim = BOTH_FORCELAND1;//BOTH_FORCEINAIR1; break; case BOTH_FORCEJUMPBACK1: newAnim = BOTH_FORCELANDBACK1;//BOTH_FORCEINAIRBACK1; break; case BOTH_FORCEJUMPLEFT1: newAnim = BOTH_FORCELANDLEFT1;//BOTH_FORCEINAIRLEFT1; break; case BOTH_FORCEJUMPRIGHT1: newAnim = BOTH_FORCELANDRIGHT1;//BOTH_FORCEINAIRRIGHT1; break; } if ( newAnim != -1 ) { int parts = SETANIM_BOTH; if ( pm->ps->weaponTime ) {//FIXME: really only care if we're in a saber attack anim... parts = SETANIM_LEGS; } PM_SetAnim( pm, parts, newAnim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); } } } */ } void PM_SwimFloatAnim( void ) { int legsAnim = pm->ps->legsAnim; //FIXME: no start or stop anims if ( pm->cmd.forwardmove || pm->cmd.rightmove || pm->cmd.upmove ) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_SWIMFORWARD,SETANIM_FLAG_NORMAL); } else {//stopping if ( legsAnim == BOTH_SWIMFORWARD ) {//I was swimming if ( !pm->ps->legsAnimTimer ) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_SWIM_IDLE1,SETANIM_FLAG_NORMAL); } } else {//idle if ( !(pm->ps->pm_flags&PMF_DUCKED) && pm->cmd.upmove >= 0 ) {//not crouching PM_SetAnim(pm,SETANIM_LEGS,BOTH_SWIM_IDLE1,SETANIM_FLAG_NORMAL); } } } } /* =============== PM_Footsteps =============== */ static void PM_Footsteps( void ) { float bobmove; int old, oldAnim; qboolean footstep = qfalse; qboolean validNPC = qfalse; qboolean flipping = qfalse; int setAnimFlags = SETANIM_FLAG_NORMAL; if( pm->gent == NULL || pm->gent->client == NULL ) return; if ( (pm->ps->eFlags&EF_HELD_BY_WAMPA) ) { PM_SetAnim( pm, SETANIM_BOTH, BOTH_HANG_IDLE, SETANIM_FLAG_NORMAL ); if ( pm->ps->legsAnim == BOTH_HANG_IDLE ) { if ( pm->ps->torsoAnimTimer < 100 ) { pm->ps->torsoAnimTimer = 100; } if ( pm->ps->legsAnimTimer < 100 ) { pm->ps->legsAnimTimer = 100; } } return; } if ( PM_SpinningSaberAnim( pm->ps->legsAnim ) && pm->ps->legsAnimTimer ) {//spinning return; } if ( PM_InKnockDown( pm->ps ) || PM_InRoll( pm->ps )) {//in knockdown return; } if ( (pm->ps->eFlags&EF_FORCE_DRAINED) ) {//being drained //PM_SetAnim( pm, SETANIM_LEGS, BOTH_HUGGEE1, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); return; } if ( (pm->ps->forcePowersActive&(1<ps->forceDrainEntityNum < ENTITYNUM_WORLD ) {//draining //PM_SetAnim( pm, SETANIM_LEGS, BOTH_HUGGER1, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); return; } else if (pm->gent->NPC && pm->gent->NPC->aiFlags & NPCAI_KNEEL) {//kneeling return; } if( pm->gent->NPC != NULL ) { validNPC = qtrue; } pm->gent->client->renderInfo.legsFpsMod = 1.0f; //PM_ResetAnkleAngles(); // // calculate speed and cycle to be used for // all cyclic walking effects // pm->xyspeed = sqrt( pm->ps->velocity[0] * pm->ps->velocity[0] + pm->ps->velocity[1] * pm->ps->velocity[1] ); if ( pm->ps->legsAnim == BOTH_FLIP_F || pm->ps->legsAnim == BOTH_FLIP_B || pm->ps->legsAnim == BOTH_FLIP_L || pm->ps->legsAnim == BOTH_FLIP_R || pm->ps->legsAnim == BOTH_ALORA_FLIP_1 || pm->ps->legsAnim == BOTH_ALORA_FLIP_2 || pm->ps->legsAnim == BOTH_ALORA_FLIP_3 ) { flipping = qtrue; } if ( pm->ps->groundEntityNum == ENTITYNUM_NONE || ( pm->watertype & CONTENTS_LADDER ) || pm->ps->waterHeightLevel >= WHL_TORSO ) {//in air or submerged in water or in ladder // airborne leaves position in cycle intact, but doesn't advance if ( pm->waterlevel > 0 ) { if ( pm->watertype & CONTENTS_LADDER ) {//FIXME: check for watertype, save waterlevel for whether to play //the get off ladder transition anim! if ( pm->ps->velocity[2] ) {//going up or down it int anim; if ( pm->ps->velocity[2] > 0 ) { anim = BOTH_LADDER_UP1; } else { anim = BOTH_LADDER_DWN1; } PM_SetAnim( pm, SETANIM_LEGS, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); if ( pm->waterlevel >= 2 ) //arms on ladder { PM_SetAnim( pm, SETANIM_TORSO, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); } if (fabs(pm->ps->velocity[2]) >5) { bobmove = 0.005 * fabs(pm->ps->velocity[2]); // climbing bobs slow if (bobmove > 0.3) bobmove = 0.3F; goto DoFootSteps; } } else { PM_SetAnim( pm, SETANIM_LEGS, BOTH_LADDER_IDLE, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD|SETANIM_FLAG_RESTART ); pm->ps->legsAnimTimer += 300; if ( pm->waterlevel >= 2 ) //arms on ladder { PM_SetAnim( pm, SETANIM_TORSO, BOTH_LADDER_IDLE, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD|SETANIM_FLAG_RESTART ); pm->ps->torsoAnimTimer += 300; } } return; } else if ( pm->ps->waterHeightLevel >= WHL_TORSO && ((pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) ||pm->ps->weapon==WP_SABER||pm->ps->weapon==WP_NONE||pm->ps->weapon==WP_MELEE) )//pm->waterlevel > 1 ) //in deep water { if ( !PM_ForceJumpingUp( pm->gent ) ) { if ( pm->ps->groundEntityNum != ENTITYNUM_NONE && (pm->ps->pm_flags&PMF_DUCKED) ) { if ( !flipping ) {//you can crouch under water if feet are on ground if ( pm->cmd.forwardmove || pm->cmd.rightmove ) { if ( pm->ps->pm_flags & PMF_BACKWARDS_RUN ) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_CROUCH1WALKBACK,setAnimFlags); } else { PM_SetAnim(pm,SETANIM_LEGS,BOTH_CROUCH1WALK,setAnimFlags); } } else { PM_SetAnim(pm,SETANIM_LEGS,BOTH_CROUCH1,SETANIM_FLAG_NORMAL); } return; } } PM_SwimFloatAnim(); if ( pm->ps->legsAnim != BOTH_SWIM_IDLE1 ) {//moving old = pm->ps->bobCycle; bobmove = 0.15f; // swim is a slow cycle pm->ps->bobCycle = (int)( old + bobmove * pml.msec ) & 255; // if we just crossed a cycle boundary, play an apropriate footstep event if ( ( ( old + 64 ) ^ ( pm->ps->bobCycle + 64 ) ) & 128 ) { PM_AddEvent( EV_SWIM ); } } } return; } else {//hmm, in water, but not high enough to swim //fall through to walk/run/stand if ( pm->ps->groundEntityNum == ENTITYNUM_NONE ) {//unless in the air //NOTE: this is a dupe of the code just below... for when you are not in the water at all if ( pm->ps->pm_flags & PMF_DUCKED ) { if ( !flipping ) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_CROUCH1,SETANIM_FLAG_NORMAL); } } else if ( pm->ps->gravity <= 0 )//FIXME: or just less than normal? { if ( pm->gent && pm->gent->client && (pm->gent->client->NPC_class == CLASS_BOBAFETT ||pm->gent->client->NPC_class == CLASS_ROCKETTROOPER) && pm->gent->client->moveType == MT_FLYSWIM ) {//flying around with jetpack //do something else? PM_JetPackAnim(); } else { PM_SwimFloatAnim(); } } return; } } } else { if ( pm->ps->pm_flags & PMF_DUCKED ) { if ( !flipping ) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_CROUCH1,SETANIM_FLAG_NORMAL); } } else if ( pm->ps->gravity <= 0 )//FIXME: or just less than normal? { if ( pm->gent && pm->gent->client && (pm->gent->client->NPC_class == CLASS_BOBAFETT||pm->gent->client->NPC_class == CLASS_ROCKETTROOPER) && pm->gent->client->moveType == MT_FLYSWIM ) {//flying around with jetpack //do something else? PM_JetPackAnim(); } else { PM_SwimFloatAnim(); } } return; } } if ( PM_SwimmingAnim( pm->ps->legsAnim ) && pm->waterlevel < 2 ) {//legs are in swim anim, and not swimming, be sure to override it setAnimFlags |= SETANIM_FLAG_OVERRIDE; } // if not trying to move if ( !pm->cmd.forwardmove && !pm->cmd.rightmove ) { if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_ATST ) { if ( !PM_AdjustStandAnimForSlope() ) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_STAND1,SETANIM_FLAG_NORMAL); } } else if ( pm->ps->pm_flags & PMF_DUCKED ) { if( !PM_InOnGroundAnim( pm->ps ) ) { if ( !PM_AdjustStandAnimForSlope() ) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_CROUCH1,SETANIM_FLAG_NORMAL); } } } else { if ( pm->ps->legsAnimTimer && PM_LandingAnim( pm->ps->legsAnim ) ) {//still in a landing anim, let it play return; } if ( pm->ps->legsAnimTimer && (pm->ps->legsAnim == BOTH_THERMAL_READY ||pm->ps->legsAnim == BOTH_THERMAL_THROW ||pm->ps->legsAnim == BOTH_ATTACK10) ) {//still in a thermal anim, let it play return; } qboolean saberInAir = qtrue; if ( pm->ps->saberInFlight ) {//guiding saber if ( PM_SaberInBrokenParry( pm->ps->saberMove ) || pm->ps->saberBlocked == BLOCKED_PARRY_BROKEN || PM_DodgeAnim( pm->ps->torsoAnim ) ) {//we're stuck in a broken parry saberInAir = qfalse; } if ( pm->ps->saberEntityNum < ENTITYNUM_NONE && pm->ps->saberEntityNum > 0 )//player is 0 {// if ( &g_entities[pm->ps->saberEntityNum] != NULL && g_entities[pm->ps->saberEntityNum].s.pos.trType == TR_STATIONARY ) {//fell to the ground and we're not trying to pull it back saberInAir = qfalse; } } } if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_GALAKMECH ) {//NOTE: stand1 is with the helmet retracted, stand1to2 is the helmet going into place PM_SetAnim( pm, SETANIM_BOTH, BOTH_STAND2, SETANIM_FLAG_NORMAL ); } else if ( pm->ps->weapon == WP_SABER && pm->ps->saberInFlight && saberInAir && (!pm->ps->dualSabers || !pm->ps->saber[1].Active())) { if ( !PM_AdjustStandAnimForSlope() ) { if ( pm->ps->legsAnim != BOTH_LOSE_SABER || !pm->ps->legsAnimTimer ) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_SABERPULL,SETANIM_FLAG_NORMAL); } } } else if ( (pm->ps->weapon == WP_SABER &&pm->ps->SaberLength()>0 &&!pm->ps->saberInFlight &&!PM_SaberDrawPutawayAnim( pm->ps->legsAnim )) ) { if ( !PM_AdjustStandAnimForSlope() ) { int legsAnim; if ( (pm->ps->torsoAnim == BOTH_SPINATTACK6 || pm->ps->torsoAnim == BOTH_SPINATTACK7 || PM_SaberInAttack( pm->ps->saberMove ) || PM_SaberInTransitionAny( pm->ps->saberMove )) && pm->ps->legsAnim != BOTH_FORCELONGLEAP_LAND && (pm->ps->groundEntityNum == ENTITYNUM_NONE//in air || (!PM_JumpingAnim( pm->ps->torsoAnim )&&!PM_InAirKickingAnim( pm->ps->torsoAnim ))) )//OR: on ground and torso not in a jump anim { legsAnim = pm->ps->torsoAnim; } else { switch ( pm->ps->saberAnimLevel ) { case SS_FAST: case SS_TAVION: legsAnim = BOTH_SABERFAST_STANCE; break; case SS_STRONG: legsAnim = BOTH_SABERSLOW_STANCE; break; case SS_DUAL: legsAnim = BOTH_SABERDUAL_STANCE; break; case SS_STAFF: legsAnim = BOTH_SABERSTAFF_STANCE; break; case SS_NONE: case SS_MEDIUM: case SS_DESANN: default: legsAnim = BOTH_STAND2; break; } } PM_SetAnim(pm,SETANIM_LEGS,legsAnim,SETANIM_FLAG_NORMAL); } } else if( (validNPC && pm->ps->weapon > WP_SABER && pm->ps->weapon < WP_DET_PACK ))//Being careful or carrying a 2-handed weapon {//Squadmates use BOTH_STAND3 oldAnim = pm->ps->legsAnim; if(oldAnim != BOTH_GUARD_LOOKAROUND1 && oldAnim != BOTH_GUARD_IDLE1 && oldAnim != BOTH_STAND2TO4 && oldAnim != BOTH_STAND4TO2 && oldAnim != BOTH_STAND4 ) {//Don't auto-override the guard idles if ( !PM_AdjustStandAnimForSlope() ) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_STAND3,SETANIM_FLAG_NORMAL); //if(oldAnim != BOTH_STAND2 && pm->ps->legsAnim == BOTH_STAND2) //{ // pm->ps->legsAnimTimer = 500; //} } } } else { if ( !PM_AdjustStandAnimForSlope() ) { // FIXME: Do we need this here... The imps stand is 4, not 1... if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_IMPERIAL ) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_STAND4,SETANIM_FLAG_NORMAL); } else if ( pm->ps->weapon == WP_TUSKEN_STAFF ) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_STAND1,SETANIM_FLAG_NORMAL); } else { if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_RANCOR ) { if ( pm->gent->count ) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_STAND4,SETANIM_FLAG_NORMAL); } else if ( pm->gent->enemy || pm->gent->wait ) {//have an enemy or have had one since we spawned PM_SetAnim(pm,SETANIM_LEGS,BOTH_STAND2,SETANIM_FLAG_NORMAL); } else { PM_SetAnim(pm,SETANIM_LEGS,BOTH_STAND1,SETANIM_FLAG_NORMAL); } } else if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_WAMPA ) { if ( pm->gent->count ) {//holding a victim PM_SetAnim(pm,SETANIM_LEGS,BOTH_HOLD_IDLE/*BOTH_STAND2*/,SETANIM_FLAG_NORMAL); } else {//not holding a victim PM_SetAnim(pm,SETANIM_LEGS,BOTH_STAND1,SETANIM_FLAG_NORMAL); } } else { PM_SetAnim(pm,SETANIM_LEGS,BOTH_STAND1,SETANIM_FLAG_NORMAL); } } } } } return; } //maybe call this every frame, even when moving? if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_ATST ) { PM_FootSlopeTrace( NULL, NULL ); } //trying to move laterally if ( (pm->ps->eFlags&EF_IN_ATST) || (pm->gent&&pm->gent->client&&pm->gent->client->NPC_class==CLASS_RANCOR)//does this catch NPCs, too? || (pm->gent&&pm->gent->client&&pm->gent->client->NPC_class==CLASS_WAMPA) )//does this catch NPCs, too? {//atst, Rancor & Wampa, only override turn anims on legs (no torso) if ( pm->ps->legsAnim == BOTH_TURN_LEFT1 || pm->ps->legsAnim == BOTH_TURN_RIGHT1 ) {//moving overrides turning setAnimFlags |= SETANIM_FLAG_OVERRIDE; } } else {//all other NPCs... if ( (PM_InSaberAnim( pm->ps->legsAnim ) && !PM_SpinningSaberAnim( pm->ps->legsAnim )) || PM_SaberStanceAnim( pm->ps->legsAnim ) || PM_SaberDrawPutawayAnim( pm->ps->legsAnim ) || pm->ps->legsAnim == BOTH_SPINATTACK6//not a full-body spin, just spinning the saber || pm->ps->legsAnim == BOTH_SPINATTACK7//not a full-body spin, just spinning the saber || pm->ps->legsAnim == BOTH_BUTTON_HOLD || pm->ps->legsAnim == BOTH_BUTTON_RELEASE || pm->ps->legsAnim == BOTH_THERMAL_READY || pm->ps->legsAnim == BOTH_THERMAL_THROW || pm->ps->legsAnim == BOTH_ATTACK10 || PM_LandingAnim( pm->ps->legsAnim ) || PM_PainAnim( pm->ps->legsAnim ) || PM_ForceAnim( pm->ps->legsAnim )) {//legs are in a saber anim, and not spinning, be sure to override it setAnimFlags |= SETANIM_FLAG_OVERRIDE; } } if ( pm->ps->pm_flags & PMF_DUCKED ) { bobmove = 0.5; // ducked characters bob much faster if( !PM_InOnGroundAnim( pm->ps ) //not on the ground && ( !PM_InRollIgnoreTimer( pm->ps )||(!pm->ps->legsAnimTimer&&pm->cmd.upmove<0) ) )//not in a roll (or you just finished one and you're still holding crouch) { qboolean rolled = qfalse; if ( PM_RunningAnim( pm->ps->legsAnim ) || pm->ps->legsAnim == BOTH_FORCEHEAL_START || PM_CanRollFromSoulCal( pm->ps )) {//roll! rolled = PM_TryRoll(); } if ( !rolled ) { if ( pm->ps->pm_flags & PMF_BACKWARDS_RUN ) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_CROUCH1WALKBACK,setAnimFlags); } else { PM_SetAnim(pm,SETANIM_LEGS,BOTH_CROUCH1WALK,setAnimFlags); } if ( !Q_irand( 0, 19 ) ) {//5% chance of making an alert AddSoundEvent( pm->gent, pm->ps->origin, 16, AEL_MINOR, qtrue, qtrue ); } } else {//rolling is a little noisy AddSoundEvent( pm->gent, pm->ps->origin, 128, AEL_MINOR, qtrue, qtrue ); } } // ducked characters never play footsteps } else if ( pm->ps->pm_flags & PMF_BACKWARDS_RUN ) {//Moving backwards if ( !( pm->cmd.buttons & BUTTON_WALKING ) ) {//running backwards bobmove = 0.4F; // faster speeds bob faster if ( pm->ps->weapon == WP_SABER && pm->ps->SaberActive() ) { /* if ( pm->ps->saberAnimLevel == SS_STAFF ) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_RUNBACK_STAFF,setAnimFlags); } else */ { PM_SetAnim(pm,SETANIM_LEGS,BOTH_RUNBACK2,setAnimFlags); } } else if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_RANCOR ) {//no run anim PM_SetAnim(pm,SETANIM_LEGS,BOTH_WALKBACK1,setAnimFlags); } else { PM_SetAnim(pm,SETANIM_LEGS,BOTH_RUNBACK1,setAnimFlags); } footstep = qtrue; } else {//walking backwards bobmove = 0.3F; // faster speeds bob faster if ( pm->ps->weapon == WP_SABER && pm->ps->SaberActive() ) { if ( pm->ps->saberAnimLevel == SS_DUAL ) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_WALKBACK_DUAL,setAnimFlags); } else if ( pm->ps->saberAnimLevel == SS_STAFF ) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_WALKBACK_STAFF,setAnimFlags); } else { PM_SetAnim(pm,SETANIM_LEGS,BOTH_WALKBACK2,setAnimFlags); } } else { PM_SetAnim(pm,SETANIM_LEGS,BOTH_WALKBACK1,setAnimFlags); } if ( !Q_irand( 0, 9 ) ) {//10% chance of a small alert, mainly for the sand_creature AddSoundEvent( pm->gent, pm->ps->origin, 16, AEL_MINOR, qtrue, qtrue ); } } } else { if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_GALAKMECH ) { bobmove = 0.3F; // walking bobs slow if ( pm->ps->weapon == WP_NONE ) {//helmet retracted PM_SetAnim( pm, SETANIM_BOTH, BOTH_WALK1, SETANIM_FLAG_NORMAL ); } else {//helmet in place PM_SetAnim( pm, SETANIM_BOTH, BOTH_WALK2, SETANIM_FLAG_NORMAL ); } AddSoundEvent( pm->gent, pm->ps->origin, 128, AEL_SUSPICIOUS, qtrue, qtrue ); } else { if ( !( pm->cmd.buttons & BUTTON_WALKING ) ) {//running bobmove = 0.4F; // faster speeds bob faster if ( pm->ps->weapon == WP_SABER && pm->ps->SaberActive() ) { if ( pm->ps->saberAnimLevel == SS_DUAL ) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_RUN_DUAL,setAnimFlags); } else if ( pm->ps->saberAnimLevel == SS_STAFF ) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_RUN_STAFF,setAnimFlags); } else { PM_SetAnim(pm,SETANIM_LEGS,BOTH_RUN2,setAnimFlags); } } else { if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_JAWA ) { //if ( pm->gent->enemy && (pm->ps->weapon == WP_NONE || pm->ps->weapon == WP_MELEE) ) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_RUN4,setAnimFlags); } /* else { PM_SetAnim(pm,SETANIM_LEGS,BOTH_RUN1,setAnimFlags); } */ } else if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_ATST ) { if ( pm->ps->legsAnim != BOTH_RUN1 ) { if ( pm->ps->legsAnim != BOTH_RUN1START ) {//Hmm, he should really start slow and have to accelerate... also need to do this for stopping PM_SetAnim( pm,SETANIM_LEGS, BOTH_RUN1START, setAnimFlags|SETANIM_FLAG_HOLD ); } else if ( !pm->ps->legsAnimTimer ) { PM_SetAnim( pm, SETANIM_LEGS, BOTH_RUN1, setAnimFlags ); } } else { PM_SetAnim( pm, SETANIM_LEGS, BOTH_RUN1, setAnimFlags ); } } else if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_WAMPA ) { if ( pm->gent->NPC && pm->gent->NPC->stats.runSpeed == 300 ) {//full on run, on all fours PM_SetAnim(pm,SETANIM_LEGS,BOTH_RUN1,SETANIM_FLAG_NORMAL); } else {//regular, upright run PM_SetAnim(pm,SETANIM_LEGS,BOTH_RUN2,SETANIM_FLAG_NORMAL); } } else if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_RANCOR ) {//no run anim PM_SetAnim( pm, SETANIM_LEGS, BOTH_WALK1, setAnimFlags ); } else { PM_SetAnim( pm, SETANIM_LEGS, BOTH_RUN1, setAnimFlags ); } } footstep = qtrue; } else {//walking forward bobmove = 0.3F; // walking bobs slow if ( pm->ps->weapon == WP_SABER && pm->ps->SaberActive() ) { if ( pm->ps->saberAnimLevel == SS_DUAL ) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_WALK_DUAL,setAnimFlags); } else if ( pm->ps->saberAnimLevel == SS_STAFF ) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_WALK_STAFF,setAnimFlags); } else { PM_SetAnim(pm,SETANIM_LEGS,BOTH_WALK2,setAnimFlags); } } else if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_WAMPA ) { if ( pm->gent->health <= 50 ) {//hurt walk PM_SetAnim(pm,SETANIM_LEGS,BOTH_WALK2,SETANIM_FLAG_NORMAL); } else {//normal walk PM_SetAnim(pm,SETANIM_LEGS,BOTH_WALK1,SETANIM_FLAG_NORMAL); } } else { PM_SetAnim(pm,SETANIM_LEGS,BOTH_WALK1,setAnimFlags); } //Enemy NPCs always make footsteps for the benefit of the player if ( pm->gent && pm->gent->NPC && pm->gent->client && pm->gent->client->playerTeam != TEAM_PLAYER ) { footstep = qtrue; } else if ( !Q_irand( 0, 9 ) ) {//10% chance of a small alert, mainly for the sand_creature AddSoundEvent( pm->gent, pm->ps->origin, 16, AEL_MINOR, qtrue, qtrue ); } } } } if(pm->gent != NULL) { if( pm->gent->client->renderInfo.legsFpsMod > 2 ) { pm->gent->client->renderInfo.legsFpsMod = 2; } else if(pm->gent->client->renderInfo.legsFpsMod < 0.5) { pm->gent->client->renderInfo.legsFpsMod = 0.5; } } DoFootSteps: // check for footstep / splash sounds old = pm->ps->bobCycle; pm->ps->bobCycle = (int)( old + bobmove * pml.msec ) & 255; // if we just crossed a cycle boundary, play an apropriate footstep event if ( ( ( old + 64 ) ^ ( pm->ps->bobCycle + 64 ) ) & 128 ) { if ( pm->watertype & CONTENTS_LADDER ) { if ( !pm->noFootsteps ) { if (pm->ps->groundEntityNum == ENTITYNUM_NONE) {// on ladder PM_AddEvent( EV_FOOTSTEP_METAL ); } else { //PM_AddEvent( PM_FootstepForSurface() ); //still on ground } } if ( pm->gent && pm->gent->s.number == 0 ) { // if ( pm->gent->client && pm->gent->client->playerTeam != TEAM_DISGUISE ) { AddSoundEvent( pm->gent, pm->ps->origin, 128, AEL_MINOR, qtrue ); } } } else if ( pm->waterlevel == 0 ) { // on ground will only play sounds if running if ( footstep ) { if ( !pm->noFootsteps ) { //PM_AddEvent( PM_FootstepForSurface() ); } if ( pm->gent && pm->gent->s.number == 0 ) { vec3_t bottom; VectorCopy( pm->ps->origin, bottom ); bottom[2] += pm->mins[2]; // if ( pm->gent->client && pm->gent->client->playerTeam != TEAM_DISGUISE ) { AddSoundEvent( pm->gent, bottom, 256, AEL_MINOR, qtrue, qtrue ); } } } } else if ( pm->waterlevel == 1 ) { // splashing if ( pm->ps->waterHeightLevel >= WHL_KNEES ) { PM_AddEvent( EV_FOOTWADE ); } else { PM_AddEvent( EV_FOOTSPLASH ); } if ( pm->gent && pm->gent->s.number == 0 ) { vec3_t bottom; VectorCopy( pm->ps->origin, bottom ); bottom[2] += pm->mins[2]; // if ( pm->gent->client && pm->gent->client->playerTeam != TEAM_DISGUISE ) { AddSoundEvent( pm->gent, pm->ps->origin, 384, AEL_SUSPICIOUS, qfalse, qtrue );//was bottom AddSightEvent( pm->gent, pm->ps->origin, 512, AEL_MINOR ); } } } else if ( pm->waterlevel == 2 ) { // wading / swimming at surface /* if ( pm->ps->waterHeightLevel >= WHL_TORSO ) { PM_AddEvent( EV_SWIM ); } else */ { PM_AddEvent( EV_FOOTWADE ); } if ( pm->gent && pm->gent->s.number == 0 ) { // if ( pm->gent->client && pm->gent->client->playerTeam != TEAM_DISGUISE ) { AddSoundEvent( pm->gent, pm->ps->origin, 256, AEL_MINOR, qfalse, qtrue ); AddSightEvent( pm->gent, pm->ps->origin, 512, AEL_SUSPICIOUS ); } } } else {// or no sound when completely underwater...? PM_AddEvent( EV_SWIM ); } } } /* ============== PM_WaterEvents Generate sound events for entering and leaving water ============== */ static void PM_WaterEvents( void ) { // FIXME? qboolean impact_splash = qfalse; if ( pm->watertype & CONTENTS_LADDER ) //fake water for ladder { return; } // // if just entered a water volume, play a sound // if (!pml.previous_waterlevel && pm->waterlevel) { if ( (pm->watertype&CONTENTS_LAVA) ) { PM_AddEvent( EV_LAVA_TOUCH ); } else { PM_AddEvent( EV_WATER_TOUCH ); } if ( pm->gent ) { if ( VectorLengthSquared( pm->ps->velocity ) > 40000 ) { impact_splash = qtrue; } if ( pm->ps->clientNum < MAX_CLIENTS ) { AddSoundEvent( pm->gent, pm->ps->origin, 384, AEL_SUSPICIOUS ); AddSightEvent( pm->gent, pm->ps->origin, 512, AEL_SUSPICIOUS ); } } } // // if just completely exited a water volume, play a sound // if (pml.previous_waterlevel && !pm->waterlevel) { if ( (pm->watertype&CONTENTS_LAVA) ) { PM_AddEvent( EV_LAVA_LEAVE ); } else { PM_AddEvent( EV_WATER_LEAVE ); } if ( pm->gent && VectorLengthSquared( pm->ps->velocity ) > 40000 ) { impact_splash = qtrue; } if ( pm->gent && pm->ps->clientNum < MAX_CLIENTS ) { AddSoundEvent( pm->gent, pm->ps->origin, 384, AEL_SUSPICIOUS ); AddSightEvent( pm->gent, pm->ps->origin, 512, AEL_SUSPICIOUS ); } } if ( impact_splash ) { //play the splash effect trace_t tr; vec3_t axis[3], angs, start, end; VectorSet( angs, 0, pm->gent->currentAngles[YAW], 0 ); AngleVectors( angs, axis[2], axis[1], axis[0] ); VectorCopy( pm->ps->origin, start ); VectorCopy( pm->ps->origin, end ); // FIXME: set start and end better start[2] += 10; end[2] -= 40; gi.trace( &tr, start, vec3_origin, vec3_origin, end, pm->gent->s.number, MASK_WATER, (EG2_Collision)0, 0 ); if ( tr.fraction < 1.0f ) { if ( (tr.contents&CONTENTS_LAVA) ) { G_PlayEffect( "env/lava_splash", tr.endpos, axis ); } else if ( (tr.contents&CONTENTS_SLIME) ) { G_PlayEffect( "env/acid_splash", tr.endpos, axis ); } else //must be water { G_PlayEffect( "env/water_impact", tr.endpos, axis ); } } } // // check for head just going under water // if (pml.previous_waterlevel != 3 && pm->waterlevel == 3) { if ( (pm->watertype&CONTENTS_LAVA) ) { PM_AddEvent( EV_LAVA_UNDER ); } else { PM_AddEvent( EV_WATER_UNDER ); } if ( pm->gent && pm->ps->clientNum < MAX_CLIENTS ) { AddSoundEvent( pm->gent, pm->ps->origin, 256, AEL_MINOR ); AddSightEvent( pm->gent, pm->ps->origin, 384, AEL_MINOR ); } } // // check for head just coming out of water // if (pml.previous_waterlevel == 3 && pm->waterlevel != 3) { if ( !pm->gent || !pm->gent->client || pm->gent->client->airOutTime < level.time + 2000 ) {//only do this if we were drowning or about to start drowning PM_AddEvent( EV_WATER_CLEAR ); } else { if ( (pm->watertype&CONTENTS_LAVA) ) { PM_AddEvent( EV_LAVA_LEAVE ); } else { PM_AddEvent( EV_WATER_LEAVE ); } } if ( pm->gent && pm->ps->clientNum < MAX_CLIENTS ) { AddSoundEvent( pm->gent, pm->ps->origin, 256, AEL_MINOR ); AddSightEvent( pm->gent, pm->ps->origin, 384, AEL_SUSPICIOUS ); } } } /* =============== PM_BeginWeaponChange =============== */ static void PM_BeginWeaponChange( int weapon ) { if ( pm->gent && pm->gent->client && pm->gent->client->pers.enterTime >= level.time - 500 ) {//just entered map if ( weapon == WP_NONE && pm->ps->weapon != weapon ) {//don't switch to weapon none if just entered map return; } } if ( weapon < WP_NONE || weapon >= WP_NUM_WEAPONS ) { return; } if ( !( pm->ps->stats[STAT_WEAPONS] & ( 1 << weapon ) ) ) { return; } if ( pm->ps->weaponstate == WEAPON_DROPPING ) { return; } if ( cg.time > 0 ) {//this way we don't get that annoying change weapon sound every time a map starts PM_AddEvent( EV_CHANGE_WEAPON ); } pm->ps->weaponstate = WEAPON_DROPPING; pm->ps->weaponTime += 200; if ( !(pm->ps->eFlags&EF_HELD_BY_WAMPA) && !G_IsRidingVehicle(pm->gent)) { PM_SetAnim(pm,SETANIM_TORSO,TORSO_DROPWEAP1,SETANIM_FLAG_HOLD); } // turn of any kind of zooming when weapon switching....except the LA Goggles // eezstreet edit: also ignore if we change to WP_NONE..sorta hacky fix for binoculars using WP_SABER if ( pm->ps->clientNum == 0 && cg.weaponSelect != WP_NONE ) { if ( cg.zoomMode > 0 && cg.zoomMode < 3 ) { cg.zoomMode = 0; cg.zoomTime = cg.time; } } if ( pm->gent && pm->gent->client && (pm->gent->client->NPC_class == CLASS_ATST||pm->gent->client->NPC_class == CLASS_RANCOR) ) { if ( pm->ps->clientNum < MAX_CLIENTS ) { gi.cvar_set( "cg_thirdperson", "1" ); } } else if ( weapon == WP_SABER ) {//going to switch to lightsaber } else { if ( pm->ps->weapon == WP_SABER ) {//going to switch away from saber if ( pm->gent ) { G_SoundOnEnt( pm->gent, CHAN_WEAPON, "sound/weapons/saber/saberoffquick.wav" ); } if (!G_IsRidingVehicle(pm->gent)) { PM_SetSaberMove(LS_PUTAWAY); } } //put this back in because saberActive isn't being set somewhere else anymore pm->ps->SaberDeactivate(); pm->ps->SetSaberLength( 0.0f ); } } /* =============== PM_FinishWeaponChange =============== */ static void PM_FinishWeaponChange( void ) { int weapon; qboolean trueSwitch = qtrue; if ( pm->gent && pm->gent->client && pm->gent->client->pers.enterTime >= level.time - 500 ) {//just entered map if ( pm->cmd.weapon == WP_NONE && pm->ps->weapon != pm->cmd.weapon ) {//don't switch to weapon none if just entered map return; } } weapon = pm->cmd.weapon; if ( weapon < WP_NONE || weapon >= WP_NUM_WEAPONS ) { weapon = WP_NONE; } if ( !( pm->ps->stats[STAT_WEAPONS] & ( 1 << weapon ) ) ) { weapon = WP_NONE; } if ( pm->ps->weapon == weapon ) { trueSwitch = qfalse; } //int oldWeap = pm->ps->weapon; pm->ps->weapon = weapon; pm->ps->weaponstate = WEAPON_RAISING; pm->ps->weaponTime += 250; if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_ATST ) {//do nothing } else if ( weapon == WP_SABER ) {//turn on the lightsaber //FIXME: somehow sometimes I still end up with 2 weapons in hand... usually if I // cycle weapons fast enough that I end up in 1st person lightsaber, then // somehow throw the saber and switch to another weapon (all in 1st person), // making my saber drop to the ground... when I switch back to the saber, it // does not remove the current weapon model and then, when I pull the saber // back to my hand, I have 2 weaponModels active...? if ( pm->gent ) {// remove gun if we had it. G_RemoveWeaponModels( pm->gent ); } if ( !pm->ps->saberInFlight || pm->ps->dualSabers ) {//if it's not in flight or lying around, turn it on! //FIXME: AddSound/Sight Event //FIXME: don't do this if just loaded a game if ( trueSwitch ) {//actually did switch weapons, turn it on if ( PM_RidingVehicle() ) {//only turn on the first saber's first blade...? pm->ps->SaberBladeActivate( 0, 0 ); } else { pm->ps->SaberActivate(); } pm->ps->SetSaberLength( 0.0f ); } if ( pm->gent ) { WP_SaberAddG2SaberModels( pm->gent ); } } else {//FIXME: pull it back to us? } if ( pm->gent ) { WP_SaberInitBladeData( pm->gent ); if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) ) { //gi.cvar_set( "cg_thirdperson", "1" ); } } if ( trueSwitch ) {//actually did switch weapons, play anim if (!G_IsRidingVehicle(pm->gent)) { PM_SetSaberMove(LS_DRAW); } } } else {//switched away from saber if ( pm->gent ) { // remove the sabre if we had it. G_RemoveWeaponModels( pm->gent ); if (weaponData[weapon].weaponMdl[0]) { //might be NONE, so check if it has a model G_CreateG2AttachedWeaponModel( pm->gent, weaponData[weapon].weaponMdl, pm->gent->handRBolt, 0 ); } } if ( !(pm->ps->eFlags&EF_HELD_BY_WAMPA) ) { if ( pm->ps->weapon != WP_THERMAL && pm->ps->weapon != WP_TRIP_MINE && pm->ps->weapon != WP_DET_PACK && !G_IsRidingVehicle(pm->gent)) { PM_SetAnim(pm,SETANIM_TORSO,TORSO_RAISEWEAP1,SETANIM_FLAG_HOLD); } } if ( pm->ps->clientNum < MAX_CLIENTS && cg_gunAutoFirst.integer && !PM_RidingVehicle() // && oldWeap == WP_SABER && weapon != WP_NONE ) { gi.cvar_set( "cg_thirdperson", "0" ); } pm->ps->saberMove = LS_NONE; pm->ps->saberBlocking = BLK_NO; pm->ps->saberBlocked = BLOCKED_NONE; } } int PM_ReadyPoseForSaberAnimLevel( void ) { int anim = BOTH_STAND2; if (PM_RidingVehicle()) { return -1; } switch ( pm->ps->saberAnimLevel ) { case SS_DUAL: anim = BOTH_SABERDUAL_STANCE; break; case SS_STAFF: anim = BOTH_SABERSTAFF_STANCE; break; case SS_FAST: case SS_TAVION: anim = BOTH_SABERFAST_STANCE; break; case SS_STRONG: anim = BOTH_SABERSLOW_STANCE; break; case SS_NONE: case SS_MEDIUM: case SS_DESANN: default: anim = BOTH_STAND2; break; } return anim; } qboolean PM_CanDoDualDoubleAttacks( void ) { if ( (pm->ps->saber[0].saberFlags&SFL_NO_MIRROR_ATTACKS) ) { return qfalse; } if ( pm->ps->dualSabers && (pm->ps->saber[1].saberFlags&SFL_NO_MIRROR_ATTACKS) ) { return qfalse; } //NOTE: assumes you're using SS_DUAL style and have both sabers on... if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) ) {//player return qtrue; } if ( pm->gent && pm->gent->NPC && pm->gent->NPC->rank >= Q_irand( RANK_LT_COMM, RANK_CAPTAIN+2 ) ) {//high-rank NPC return qtrue; } if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_ALORA ) {//Alora return qtrue; } return qfalse; } void PM_SetJumped( float height, qboolean force ) { pm->ps->velocity[2] = height; pml.groundPlane = qfalse; pml.walking = qfalse; pm->ps->groundEntityNum = ENTITYNUM_NONE; pm->ps->pm_flags |= PMF_JUMP_HELD; pm->ps->pm_flags |= PMF_JUMPING; pm->cmd.upmove = 0; if ( force ) { pm->ps->jumpZStart = pm->ps->origin[2]; pm->ps->pm_flags |= PMF_SLOW_MO_FALL; //start force jump pm->ps->forcePowersActive |= (1<gent, CHAN_BODY, "sound/weapons/force/jump.wav" ); } else { PM_AddEvent( EV_JUMP ); } } void PM_SetSaberMove(saberMoveName_t newMove) { unsigned int setflags; int anim; int parts = SETANIM_TORSO; qboolean manualBlocking = qfalse; if ( newMove < LS_NONE || newMove >= LS_MOVE_MAX ) { assert(0); return; } setflags = saberMoveData[newMove].animSetFlags; anim = saberMoveData[newMove].animToUse; if ( (pm->ps->eFlags&EF_HELD_BY_WAMPA) ) {//no anim return; } if ( cg_debugSaber.integer&0x01 && (newMove != LS_READY) ) { Com_Printf("SetSaberMove: From '%s' to '%s'\n", saberMoveData[pm->ps->saberMove].name, saberMoveData[newMove].name); } if ( newMove == LS_READY || newMove == LS_A_FLIP_STAB || newMove == LS_A_FLIP_SLASH ) {//finished with a kata (or in a special move) reset attack counter pm->ps->saberAttackChainCount = 0; } else if ( PM_SaberInAttack( newMove ) ) {//continuing with a kata, increment attack counter //FIXME: maybe some contextual/style-specific logic in here pm->ps->saberAttackChainCount++; } if ( newMove == LS_READY ) { if ( pm->ps->saberBlockingTime > cg.time ) { manualBlocking = qtrue; if ( !pm->ps->SaberActive() ) {//turn on all blades and sabers if none are currently on pm->ps->SaberActivate(); } if ( pm->ps->saber[0].type == SABER_CLAW ) { anim = BOTH_INAIR1;//FIXME: is there a better anim for this? } else if ( pm->ps->dualSabers && pm->ps->saber[1].Active() ) { anim = BOTH_INAIR1; } else { anim = BOTH_P1_S1_T_; } } else if ( pm->ps->saber[0].readyAnim != -1 ) { anim = pm->ps->saber[0].readyAnim; } else if ( pm->ps->dualSabers && pm->ps->saber[1].readyAnim != -1 ) { anim = pm->ps->saber[1].readyAnim; } else if ( pm->ps->saber[0].type == SABER_ARC ) {//FIXME: need it's own style? anim = BOTH_SABERFAST_STANCE; } else if ( (pm->ps->dualSabers && pm->ps->saber[1].Active()) ) { anim = BOTH_SABERDUAL_STANCE; } else if ( (pm->ps->SaberStaff() && (!pm->ps->saber[0].singleBladeStyle||pm->ps->saber[0].blade[1].active))//saber staff with more than first blade active || pm->ps->saber[0].type == SABER_ARC ) { anim = BOTH_SABERSTAFF_STANCE; } else if ( pm->ps->saber[0].type == SABER_LANCE || pm->ps->saber[0].type == SABER_TRIDENT ) {//FIXME: need some 2-handed forward-pointing anim anim = BOTH_STAND1; } else { anim = PM_ReadyPoseForSaberAnimLevel(); } } else if ( newMove == LS_DRAW ) { if ( PM_RunningAnim( pm->ps->torsoAnim ) ) { pm->ps->saberMove = newMove; return; } if ( pm->ps->saber[0].drawAnim != -1 ) { anim = pm->ps->saber[0].drawAnim; } else if ( pm->ps->dualSabers && pm->ps->saber[1].drawAnim != -1 ) { anim = pm->ps->saber[1].drawAnim; } else if ( pm->ps->saber[0].stylesLearned==(1<ps->dualSabers && !(pm->ps->saber[0].stylesForbidden&(1<ps->saber[1].stylesForbidden&(1<ps->torsoAnim == BOTH_STAND1IDLE1 ) { setflags |= SETANIM_FLAG_OVERRIDE; } } else if ( newMove == LS_PUTAWAY ) { if ( pm->ps->saber[0].putawayAnim != -1 ) { anim = pm->ps->saber[0].putawayAnim; } else if ( pm->ps->dualSabers && pm->ps->saber[1].putawayAnim != -1 ) { anim = pm->ps->saber[1].putawayAnim; } else if ( pm->ps->saber[0].stylesLearned==(1<ps->saber[0].blade[1].active ) { anim = BOTH_S7_S1; } else if ( pm->ps->dualSabers && !(pm->ps->saber[0].stylesForbidden&(1<ps->saber[1].stylesForbidden&(1<ps->saber[1].Active() ) { anim = BOTH_S6_S1; } if ( PM_SaberStanceAnim( pm->ps->legsAnim ) && pm->ps->legsAnim != BOTH_STAND1 ) { parts = SETANIM_BOTH; } else { if ( PM_RunningAnim( pm->ps->torsoAnim ) ) { pm->ps->saberMove = newMove; return; } parts = SETANIM_TORSO; } //FIXME: also dual } else if ( pm->ps->saberAnimLevel == SS_STAFF && newMove >= LS_S_TL2BR && newMove < LS_REFLECT_LL ) {//staff has an entirely new set of anims, besides special attacks //FIXME: include ready and draw/putaway? //FIXME: get hand-made bounces and deflections? if ( newMove >= LS_V1_BR && newMove <= LS_REFLECT_LL ) {//there aren't 1-7, just 1, 6 and 7, so just set it anim = BOTH_P7_S7_T_ + (anim-BOTH_P1_S1_T_);//shift it up to the proper set } else {//add the appropriate animLevel anim += (pm->ps->saberAnimLevel-FORCE_LEVEL_1) * SABER_ANIM_GROUP_SIZE; } } else if ( (pm->ps->saberAnimLevel == SS_DUAL || (pm->ps->dualSabers&& pm->ps->saber[1].Active())) && newMove >= LS_S_TL2BR && newMove < LS_REFLECT_LL ) {//staff has an entirely new set of anims, besides special attacks //FIXME: include ready and draw/putaway? //FIXME: get hand-made bounces and deflections? //FIXME: only do the dual FB & LR attacks when on ground? if ( newMove >= LS_V1_BR && newMove <= LS_REFLECT_LL ) {//there aren't 1-7, just 1, 6 and 7, so just set it anim = BOTH_P6_S6_T_ + (anim-BOTH_P1_S1_T_);//shift it up to the proper set } else if ( ( newMove == LS_A_R2L || newMove == LS_S_R2L || newMove == LS_A_L2R || newMove == LS_S_L2R ) && PM_CanDoDualDoubleAttacks() && G_CheckEnemyPresence( pm->gent, DIR_RIGHT, 150.0f ) && G_CheckEnemyPresence( pm->gent, DIR_LEFT, 150.0f ) ) {//enemy both on left and right newMove = LS_DUAL_LR; anim = saberMoveData[newMove].animToUse; //probably already moved, but... pm->cmd.rightmove = 0; } else if ( (newMove == LS_A_T2B || newMove == LS_S_T2B || newMove == LS_A_BACK || newMove == LS_A_BACK_CR ) && PM_CanDoDualDoubleAttacks() && G_CheckEnemyPresence( pm->gent, DIR_FRONT, 150.0f ) && G_CheckEnemyPresence( pm->gent, DIR_BACK, 150.0f ) ) {//enemy both in front and back newMove = LS_DUAL_FB; anim = saberMoveData[newMove].animToUse; //probably already moved, but... pm->cmd.forwardmove = 0; } else {//add the appropriate animLevel anim += (pm->ps->saberAnimLevel-FORCE_LEVEL_1) * SABER_ANIM_GROUP_SIZE; } } /* else if ( newMove == LS_DRAW && pm->ps->saberAnimLevel == SS_STAFF )//pm->ps->SaberStaff() ) {//hold saber out front as we turn it on //FIXME: need a real "draw" anim for this (and put-away) anim = BOTH_SABERSTAFF_STANCE; } */ else if ( pm->ps->saberAnimLevel > FORCE_LEVEL_1 && !PM_SaberInIdle( newMove ) && !PM_SaberInParry( newMove ) && !PM_SaberInKnockaway( newMove ) && !PM_SaberInBrokenParry( newMove ) && !PM_SaberInReflect( newMove ) && !PM_SaberInSpecial( newMove )) {//readies, parries and reflections have only 1 level if ( pm->ps->saber[0].type == SABER_LANCE || pm->ps->saber[0].type == SABER_TRIDENT ) {//FIXME: hack for now - these use the fast anims, but slowed down. Should have own style } else {//increment the anim to the next level of saber anims anim += (pm->ps->saberAnimLevel-FORCE_LEVEL_1) * SABER_ANIM_GROUP_SIZE; } } else if ( newMove == LS_KICK_F_AIR || newMove == LS_KICK_B_AIR || newMove == LS_KICK_R_AIR || newMove == LS_KICK_L_AIR ) { if ( pm->ps->groundEntityNum != ENTITYNUM_NONE ) { PM_SetJumped( 200, qtrue ); } } // If the move does the same animation as the last one, we need to force a restart... // if ( saberMoveData[pm->ps->saberMove].animToUse == anim && newMove > LS_PUTAWAY) if ( ( pm->ps->torsoAnim == anim || pm->ps->legsAnim == anim ) && newMove > LS_PUTAWAY ) { setflags |= SETANIM_FLAG_RESTART; } if ( (anim == BOTH_STAND1 && (pm->ps->saber[0].type == SABER_ARC || (pm->ps->dualSabers && pm->ps->saber[1].Active())) ) || anim == BOTH_STAND2 //FIXME: temp hack to stop it from using run2 with staff || (0 && anim == BOTH_SABERSTAFF_STANCE) || anim == BOTH_SABERDUAL_STANCE || anim == BOTH_SABERFAST_STANCE || anim == BOTH_SABERSLOW_STANCE ) {//match torso anim to walk/run anim if newMove is just LS_READY //FIXME: play both_stand2_random1 when you've been idle for a while switch ( pm->ps->legsAnim ) { case BOTH_WALK1: case BOTH_WALK2: case BOTH_WALK_STAFF: case BOTH_WALK_DUAL: case BOTH_WALKBACK1: case BOTH_WALKBACK2: case BOTH_WALKBACK_STAFF: case BOTH_WALKBACK_DUAL: case BOTH_RUN1: case BOTH_RUN2: case BOTH_RUN_STAFF: case BOTH_RUN_DUAL: case BOTH_RUNBACK1: case BOTH_RUNBACK2: case BOTH_RUNBACK_STAFF: anim = pm->ps->legsAnim; break; } } if ( !PM_RidingVehicle() ) { if ( !manualBlocking ) { if ( newMove == LS_A_LUNGE || newMove == LS_A_JUMP_T__B_ || newMove == LS_A_BACKSTAB || newMove == LS_A_BACK || newMove == LS_A_BACK_CR || newMove == LS_ROLL_STAB || newMove == LS_A_FLIP_STAB || newMove == LS_A_FLIP_SLASH || newMove == LS_JUMPATTACK_DUAL || newMove == LS_JUMPATTACK_ARIAL_LEFT || newMove == LS_JUMPATTACK_ARIAL_RIGHT || newMove == LS_JUMPATTACK_CART_LEFT || newMove == LS_JUMPATTACK_CART_RIGHT || newMove == LS_JUMPATTACK_STAFF_LEFT || newMove == LS_JUMPATTACK_STAFF_RIGHT || newMove == LS_BUTTERFLY_LEFT || newMove == LS_BUTTERFLY_RIGHT || newMove == LS_A_BACKFLIP_ATK || newMove == LS_STABDOWN || newMove == LS_STABDOWN_STAFF || newMove == LS_STABDOWN_DUAL || newMove == LS_DUAL_SPIN_PROTECT || newMove == LS_STAFF_SOULCAL || newMove == LS_A1_SPECIAL || newMove == LS_A2_SPECIAL || newMove == LS_A3_SPECIAL || newMove == LS_UPSIDE_DOWN_ATTACK || newMove == LS_PULL_ATTACK_STAB || newMove == LS_PULL_ATTACK_SWING || PM_KickMove( newMove ) ) { parts = SETANIM_BOTH; } else if ( PM_SpinningSaberAnim( anim ) ) {//spins must be played on entire body parts = SETANIM_BOTH; } else if ( (!pm->cmd.forwardmove&&!pm->cmd.rightmove&&!pm->cmd.upmove)) {//not trying to run, duck or jump if ( !PM_FlippingAnim( pm->ps->legsAnim ) && !PM_InRoll( pm->ps ) && !PM_InKnockDown( pm->ps ) && !PM_JumpingAnim( pm->ps->legsAnim ) && !PM_PainAnim( pm->ps->legsAnim ) && !PM_InSpecialJump( pm->ps->legsAnim ) && !PM_InSlopeAnim( pm->ps->legsAnim ) && //!PM_CrouchAnim( pm->ps->legsAnim ) && //pm->cmd.upmove >= 0 && !(pm->ps->pm_flags & PMF_DUCKED) && newMove != LS_PUTAWAY ) { parts = SETANIM_BOTH; } else if ( !(pm->ps->pm_flags & PMF_DUCKED) && ( newMove == LS_SPINATTACK_DUAL || newMove == LS_SPINATTACK ) ) { parts = SETANIM_BOTH; } } } } else { if (!pm->ps->saberBlocked) { parts = SETANIM_BOTH; setflags &= ~SETANIM_FLAG_RESTART; } } if (anim!=-1) { PM_SetAnim( pm, parts, anim, setflags, saberMoveData[newMove].blendTime ); } if ( pm->ps->torsoAnim == anim ) {//successfully changed anims //special check for *starting* a saber swing if ( pm->gent && pm->ps->SaberLength() > 1 ) { if ( PM_SaberInAttack( newMove ) || PM_SaberInSpecialAttack( anim ) ) {//playing an attack if ( pm->ps->saberMove != newMove ) {//wasn't playing that attack before if ( PM_SaberInSpecialAttack( anim ) ) { WP_SaberSwingSound( pm->gent, 0, SWING_FAST ); if ( !PM_InCartwheel( pm->ps->torsoAnim ) ) {//can still attack during a cartwheel/arial pm->ps->weaponTime = pm->ps->torsoAnimTimer;//so we know our weapon is busy } } else { switch ( pm->ps->saberAnimLevel ) { case SS_DESANN: case SS_STRONG: WP_SaberSwingSound( pm->gent, 0, SWING_STRONG ); break; case SS_MEDIUM: case SS_DUAL: case SS_STAFF: WP_SaberSwingSound( pm->gent, 0, SWING_MEDIUM ); break; case SS_TAVION: case SS_FAST: WP_SaberSwingSound( pm->gent, 0, SWING_FAST ); break; } } } else if ( (setflags&SETANIM_FLAG_RESTART) && PM_SaberInSpecialAttack( anim ) ) {//sigh, if restarted a special, then set the weaponTime *again* if ( !PM_InCartwheel( pm->ps->torsoAnim ) ) {//can still attack during a cartwheel/arial pm->ps->weaponTime = pm->ps->torsoAnimTimer;//so we know our weapon is busy } } } else if ( PM_SaberInStart( newMove ) ) { //if ( g_saberRealisticCombat->integer < 1 ) {//don't damage on the first few frames of a start anim because it may pop from one position to some drastically different one, killing the enemy without hitting them. int damageDelay = 150; if ( pm->ps->torsoAnimTimer < damageDelay ) { damageDelay = pm->ps->torsoAnimTimer; } //ent->client->ps.saberDamageDebounceTime = level.time + damageDelay; } if ( pm->ps->saberAnimLevel == SS_STRONG ) { WP_SaberSwingSound( pm->gent, 0, SWING_FAST ); } } } /* //wtf... getting stuck with weaponTime set even though we're not in an attack...? if ( PM_SaberInAttack( pm->ps->saberMove ) && !PM_SaberInAttack( newMove ) ) { pm->ps->weaponTime = 0; } */ //Some special attacks can be started when sabers are off, make sure we turn them on, first! switch ( newMove ) {//make sure the saber is on! case LS_A_LUNGE: case LS_ROLL_STAB: if ( PM_InSecondaryStyle() ) {//staff as medium or dual as fast if ( pm->ps->dualSabers ) {//only force on the first saber pm->ps->saber[0].Activate(); } else if ( pm->ps->saber[0].numBlades > 1 ) {//only force on the first saber's first blade pm->ps->SaberBladeActivate(0,0); } } else {//turn on all blades on all sabers pm->ps->SaberActivate(); } break; case LS_SPINATTACK_ALORA: case LS_SPINATTACK_DUAL: case LS_SPINATTACK: case LS_A1_SPECIAL: case LS_A2_SPECIAL: case LS_A3_SPECIAL: case LS_DUAL_SPIN_PROTECT: case LS_STAFF_SOULCAL: //FIXME: probably more... pm->ps->SaberActivate(); break; default: break; } pm->ps->saberMove = newMove; pm->ps->saberBlocking = saberMoveData[newMove].blocking; if ( pm->ps->clientNum == 0 || PM_ControlledByPlayer() ) { if ( pm->ps->saberBlocked >= BLOCKED_UPPER_RIGHT_PROJ && pm->ps->saberBlocked <= BLOCKED_TOP_PROJ && newMove >= LS_REFLECT_UP && newMove <= LS_REFLECT_LL ) {//don't clear it when blocking projectiles } else { pm->ps->saberBlocked = BLOCKED_NONE; } } else if ( pm->ps->saberBlocked <= BLOCKED_ATK_BOUNCE || !pm->ps->SaberActive() || (newMove < LS_PARRY_UR || newMove > LS_REFLECT_LL) ) {//NPCs only clear blocked if not blocking? pm->ps->saberBlocked = BLOCKED_NONE; } if ( pm->gent && pm->gent->client ) { if ( saberMoveData[newMove].trailLength > 0 ) { pm->gent->client->ps.SaberActivateTrail( saberMoveData[newMove].trailLength ); // saber trail lasts for 75ms...feel free to change this if you want it longer or shorter } else { pm->gent->client->ps.SaberDeactivateTrail( 0 ); } } } } /* ============== PM_Use Generates a use event ============== */ #define USE_DELAY 250 void PM_Use( void ) { if ( pm->ps->useTime > 0 ) { pm->ps->useTime -= pml.msec; if ( pm->ps->useTime < 0 ) { pm->ps->useTime = 0; } } if ( pm->ps->useTime > 0 ) { return; } if ( ! (pm->cmd.buttons & BUTTON_USE ) ) { pm->useEvent = 0; pm->ps->useTime = 0; return; } pm->useEvent = EV_USE; pm->ps->useTime = USE_DELAY; } extern saberMoveName_t PM_AttackForEnemyPos( qboolean allowFB, qboolean allowStabDown ); saberMoveName_t PM_NPCSaberAttackFromQuad( int quad ) { //FIXME: this should be an AI decision // It should be based on the enemy's current LS_ move, saberAnimLevel, // the jedi's difficulty level, rank and FP_OFFENSE skill... saberMoveName_t autoMove = LS_NONE; if ( pm->gent && ((pm->gent->NPC && pm->gent->NPC->rank != RANK_ENSIGN && pm->gent->NPC->rank != RANK_CIVILIAN ) || (pm->gent->client && (pm->gent->client->NPC_class == CLASS_TAVION||pm->gent->client->NPC_class == CLASS_ALORA))) ) { autoMove = PM_AttackForEnemyPos( qtrue, qtrue ); } if ( autoMove != LS_NONE && PM_SaberInSpecial( autoMove ) ) {//if have opportunity to do a special attack, do one return autoMove; } else {//pick another one saberMoveName_t newmove = LS_NONE; switch( quad ) { case Q_T://blocked top if ( Q_irand( 0, 1 ) ) { newmove = LS_A_T2B; } else { newmove = LS_A_TR2BL; } break; case Q_TR: if ( !Q_irand( 0, 2 ) ) { newmove = LS_A_R2L; } else if ( !Q_irand( 0, 1 ) ) { newmove = LS_A_TR2BL; } else { newmove = LS_T1_TR_BR; } break; case Q_TL: if ( !Q_irand( 0, 2 ) ) { newmove = LS_A_L2R; } else if ( !Q_irand( 0, 1 ) ) { newmove = LS_A_TL2BR; } else { newmove = LS_T1_TL_BL; } break; case Q_BR: if ( !Q_irand( 0, 2 ) ) { newmove = LS_A_BR2TL; } else if ( !Q_irand( 0, 1 ) ) { newmove = LS_T1_BR_TR; } else { newmove = LS_A_R2L; } break; case Q_BL: if ( !Q_irand( 0, 2 ) ) { newmove = LS_A_BL2TR; } else if ( !Q_irand( 0, 1 ) ) { newmove = LS_T1_BL_TL; } else { newmove = LS_A_L2R; } break; case Q_L: if ( !Q_irand( 0, 2 ) ) { newmove = LS_A_L2R; } else if ( !Q_irand( 0, 1 ) ) { newmove = LS_T1__L_T_; } else { newmove = LS_A_R2L; } break; case Q_R: if ( !Q_irand( 0, 2 ) ) { newmove = LS_A_R2L; } else if ( !Q_irand( 0, 1 ) ) { newmove = LS_T1__R_T_; } else { newmove = LS_A_L2R; } break; case Q_B: if ( pm->gent && pm->gent->NPC && pm->gent->NPC->rank >= RANK_LT_JG ) {//fencers and above can do bottom-up attack if ( Q_irand( 0, pm->gent->NPC->rank ) >= RANK_LT_JG ) {//but not overly likely newmove = PM_SaberLungeAttackMove( qtrue ); } } break; default: break; } return newmove; } } int PM_SaberMoveQuadrantForMovement( usercmd_t *ucmd ) { if ( ucmd->rightmove > 0 ) {//moving right if ( ucmd->forwardmove > 0 ) {//forward right = TL2BR slash return Q_TL; } else if ( ucmd->forwardmove < 0 ) {//backward right = BL2TR uppercut return Q_BL; } else {//just right is a left slice return Q_L; } } else if ( ucmd->rightmove < 0 ) {//moving left if ( ucmd->forwardmove > 0 ) {//forward left = TR2BL slash return Q_TR; } else if ( ucmd->forwardmove < 0 ) {//backward left = BR2TL uppercut return Q_BR; } else {//just left is a right slice return Q_R; } } else {//not moving left or right if ( ucmd->forwardmove > 0 ) {//forward= T2B slash return Q_T; } else if ( ucmd->forwardmove < 0 ) {//backward= T2B slash //or B2T uppercut? return Q_T; } else //if ( curmove == LS_READY )//??? {//Not moving at all return Q_R; } } //return Q_R;//???? } void PM_SetAnimFrame( gentity_t *gent, int frame, qboolean torso, qboolean legs ) { if ( !gi.G2API_HaveWeGhoul2Models( gent->ghoul2 ) ) { return; } int actualTime = (cg.time?cg.time:level.time); if ( torso && gent->lowerLumbarBone != -1 )//gent->upperLumbarBone { gi.G2API_SetBoneAnimIndex(&gent->ghoul2[gent->playerModel], gent->lowerLumbarBone, //gent->upperLumbarBone frame, frame+1, BONE_ANIM_OVERRIDE_FREEZE|BONE_ANIM_BLEND, 1, actualTime, frame, 150 ); if ( gent->motionBone != -1 ) { gi.G2API_SetBoneAnimIndex(&gent->ghoul2[gent->playerModel], gent->motionBone, //gent->upperLumbarBone frame, frame+1, BONE_ANIM_OVERRIDE_FREEZE|BONE_ANIM_BLEND, 1, actualTime, frame, 150 ); } } if ( legs && gent->rootBone != -1 ) { gi.G2API_SetBoneAnimIndex(&gent->ghoul2[gent->playerModel], gent->rootBone, frame, frame+1, BONE_ANIM_OVERRIDE_FREEZE|BONE_ANIM_BLEND, 1, actualTime, frame, 150 ); } } int PM_SaberLockWinAnim( saberLockResult_t result, int breakType ) { int winAnim = -1; switch ( pm->ps->torsoAnim ) { /* default: #ifndef FINAL_BUILD Com_Printf( S_COLOR_RED"ERROR-PM_SaberLockBreak: %s not in saberlock anim, anim = (%d)%s\n", pm->gent->NPC_type, pm->ps->torsoAnim, animTable[pm->ps->torsoAnim].name ); #endif */ case BOTH_BF2LOCK: if ( breakType == SABERLOCK_SUPERBREAK ) { winAnim = BOTH_LK_S_S_T_SB_1_W; } else if ( result == LOCK_DRAW ) { winAnim = BOTH_BF1BREAK; } else { pm->ps->saberMove = LS_A_T2B; winAnim = BOTH_A3_T__B_; } break; case BOTH_BF1LOCK: if ( breakType == SABERLOCK_SUPERBREAK ) { winAnim = BOTH_LK_S_S_T_SB_1_W; } else if ( result == LOCK_DRAW ) { winAnim = BOTH_KNOCKDOWN4; } else { pm->ps->saberMove = LS_K1_T_; winAnim = BOTH_K1_S1_T_; } break; case BOTH_CWCIRCLELOCK: if ( breakType == SABERLOCK_SUPERBREAK ) { winAnim = BOTH_LK_S_S_S_SB_1_W; } else if ( result == LOCK_DRAW ) { pm->ps->saberMove = pm->ps->saberBounceMove = LS_V1_BL; pm->ps->saberBlocked = BLOCKED_PARRY_BROKEN; winAnim = BOTH_V1_BL_S1; } else { winAnim = BOTH_CWCIRCLEBREAK; } break; case BOTH_CCWCIRCLELOCK: if ( breakType == SABERLOCK_SUPERBREAK ) { winAnim = BOTH_LK_S_S_S_SB_1_W; } else if ( result == LOCK_DRAW ) { pm->ps->saberMove = pm->ps->saberBounceMove = LS_V1_BR; pm->ps->saberBlocked = BLOCKED_PARRY_BROKEN; winAnim = BOTH_V1_BR_S1; } else { winAnim = BOTH_CCWCIRCLEBREAK; } break; default: //must be using new system: break; } if ( winAnim != -1 ) { PM_SetAnim( pm, SETANIM_BOTH, winAnim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); pm->ps->weaponTime = pm->ps->torsoAnimTimer; pm->ps->saberBlocked = BLOCKED_NONE; pm->ps->weaponstate = WEAPON_FIRING; if ( breakType == SABERLOCK_SUPERBREAK && winAnim != BOTH_LK_ST_DL_T_SB_1_W ) {//going to attack with saber, do a saber trail pm->ps->SaberActivateTrail( 200 ); } } return winAnim; } int PM_SaberLockLoseAnim( gentity_t *genemy, saberLockResult_t result, int breakType ) { int loseAnim = -1; switch ( genemy->client->ps.torsoAnim ) { /* default: #ifndef FINAL_BUILD Com_Printf( S_COLOR_RED"ERROR-PM_SaberLockBreak: %s not in saberlock anim, anim = (%d)%s\n", genemy->NPC_type, genemy->client->ps.torsoAnim, animTable[genemy->client->ps.torsoAnim].name ); #endif */ case BOTH_BF2LOCK: if ( breakType == SABERLOCK_SUPERBREAK ) { loseAnim = BOTH_LK_S_S_T_SB_1_L; } else if ( result == LOCK_DRAW ) { loseAnim = BOTH_BF1BREAK; } else { if ( result == LOCK_STALEMATE ) {//no-one won genemy->client->ps.saberMove = LS_K1_T_; loseAnim = BOTH_K1_S1_T_; } else {//FIXME: this anim needs to transition back to ready when done loseAnim = BOTH_BF1BREAK; } } break; case BOTH_BF1LOCK: if ( breakType == SABERLOCK_SUPERBREAK ) { loseAnim = BOTH_LK_S_S_T_SB_1_L; } else if ( result == LOCK_DRAW ) { loseAnim = BOTH_KNOCKDOWN4; } else { if ( result == LOCK_STALEMATE ) {//no-one won genemy->client->ps.saberMove = LS_A_T2B; loseAnim = BOTH_A3_T__B_; } else { loseAnim = BOTH_KNOCKDOWN4; } } break; case BOTH_CWCIRCLELOCK: if ( breakType == SABERLOCK_SUPERBREAK ) { loseAnim = BOTH_LK_S_S_S_SB_1_L; } else if ( result == LOCK_DRAW ) { genemy->client->ps.saberMove = genemy->client->ps.saberBounceMove = LS_V1_BL; genemy->client->ps.saberBlocked = BLOCKED_PARRY_BROKEN; loseAnim = BOTH_V1_BL_S1; } else { if ( result == LOCK_STALEMATE ) {//no-one won loseAnim = BOTH_CCWCIRCLEBREAK; } else { genemy->client->ps.saberMove = genemy->client->ps.saberBounceMove = LS_V1_BL; genemy->client->ps.saberBlocked = BLOCKED_PARRY_BROKEN; loseAnim = BOTH_V1_BL_S1; /* genemy->client->ps.saberMove = genemy->client->ps.saberBounceMove = LS_H1_BR; genemy->client->ps.saberBlocked = BLOCKED_PARRY_BROKEN; loseAnim = BOTH_H1_S1_BL; */ } } break; case BOTH_CCWCIRCLELOCK: if ( breakType == SABERLOCK_SUPERBREAK ) { loseAnim = BOTH_LK_S_S_S_SB_1_L; } else if ( result == LOCK_DRAW ) { genemy->client->ps.saberMove = genemy->client->ps.saberBounceMove = LS_V1_BR; genemy->client->ps.saberBlocked = BLOCKED_PARRY_BROKEN; loseAnim = BOTH_V1_BR_S1; } else { if ( result == LOCK_STALEMATE ) {//no-one won loseAnim = BOTH_CWCIRCLEBREAK; } else { genemy->client->ps.saberMove = genemy->client->ps.saberBounceMove = LS_V1_BR; genemy->client->ps.saberBlocked = BLOCKED_PARRY_BROKEN; loseAnim = BOTH_V1_BR_S1; /* genemy->client->ps.saberMove = genemy->client->ps.saberBounceMove = LS_H1_BL; genemy->client->ps.saberBlocked = BLOCKED_PARRY_BROKEN; loseAnim = BOTH_H1_S1_BR; */ } } break; } if ( loseAnim != -1 ) { NPC_SetAnim( genemy, SETANIM_BOTH, loseAnim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); genemy->client->ps.weaponTime = genemy->client->ps.torsoAnimTimer;// + 250; genemy->client->ps.saberBlocked = BLOCKED_NONE; genemy->client->ps.weaponstate = WEAPON_READY; } return loseAnim; } int PM_SaberLockResultAnim( gentity_t *duelist, int lockOrBreakOrSuperBreak, int winOrLose ) { int baseAnim = duelist->client->ps.torsoAnim; switch ( baseAnim ) { case BOTH_LK_S_S_S_L_2: //lock if I'm using single vs. a single and other intitiated baseAnim = BOTH_LK_S_S_S_L_1; break; case BOTH_LK_S_S_T_L_2: //lock if I'm using single vs. a single and other initiated baseAnim = BOTH_LK_S_S_T_L_1; break; case BOTH_LK_DL_DL_S_L_2: //lock if I'm using dual vs. dual and other initiated baseAnim = BOTH_LK_DL_DL_S_L_1; break; case BOTH_LK_DL_DL_T_L_2: //lock if I'm using dual vs. dual and other initiated baseAnim = BOTH_LK_DL_DL_T_L_1; break; case BOTH_LK_ST_ST_S_L_2: //lock if I'm using staff vs. a staff and other initiated baseAnim = BOTH_LK_ST_ST_S_L_1; break; case BOTH_LK_ST_ST_T_L_2: //lock if I'm using staff vs. a staff and other initiated baseAnim = BOTH_LK_ST_ST_T_L_1; break; } //what kind of break? if ( lockOrBreakOrSuperBreak == SABERLOCK_BREAK ) { baseAnim -= 2; } else if ( lockOrBreakOrSuperBreak == SABERLOCK_SUPERBREAK ) { baseAnim += 1; } else {//WTF? Not a valid result return -1; } //win or lose? if ( winOrLose == SABERLOCK_WIN ) { baseAnim += 1; } //play the anim and hold it NPC_SetAnim( duelist, SETANIM_BOTH, baseAnim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); if ( lockOrBreakOrSuperBreak == SABERLOCK_SUPERBREAK && winOrLose == SABERLOCK_LOSE ) {//if you lose a superbreak, you're defenseless //make saberent not block gentity_t *saberent = &g_entities[duelist->client->ps.saberEntityNum]; if ( saberent ) { VectorClear(saberent->mins); VectorClear(saberent->maxs); G_SetOrigin(saberent, duelist->currentOrigin); } //set sabermove to none duelist->client->ps.saberMove = LS_NONE; //Hold the anim a little longer than it is duelist->client->ps.torsoAnimTimer += 250;; } //no attacking during this anim duelist->client->ps.weaponTime = duelist->client->ps.torsoAnimTimer; duelist->client->ps.saberBlocked = BLOCKED_NONE; if ( lockOrBreakOrSuperBreak == SABERLOCK_SUPERBREAK && winOrLose == SABERLOCK_WIN && baseAnim != BOTH_LK_ST_DL_T_SB_1_W ) {//going to attack with saber, do a saber trail duelist->client->ps.SaberActivateTrail( 200 ); } return baseAnim; } void PM_SaberLockBreak( gentity_t *gent, gentity_t *genemy, saberLockResult_t result, int victoryStrength ) { int winAnim = -1, loseAnim = -1; int breakType = SABERLOCK_BREAK; qboolean singleVsSingle = qtrue; if ( result == LOCK_VICTORY && Q_irand(0,7) < victoryStrength ) { if ( genemy && genemy->NPC && ((genemy->NPC->aiFlags&NPCAI_BOSS_CHARACTER) ||(genemy->NPC->aiFlags&NPCAI_SUBBOSS_CHARACTER) ||(genemy->client&&genemy->client->NPC_class == CLASS_SHADOWTROOPER)) && Q_irand(0, 4) ) {//less of a chance of getting a superbreak against a boss breakType = SABERLOCK_BREAK; } else { breakType = SABERLOCK_SUPERBREAK; } } else { breakType = SABERLOCK_BREAK; } winAnim = PM_SaberLockWinAnim( result, breakType ); if ( winAnim != -1 ) {//a single vs. single break if ( genemy && genemy->client ) { loseAnim = PM_SaberLockLoseAnim( genemy, result, breakType ); } } else {//must be a saberlock that's not between single and single... singleVsSingle = qfalse; winAnim = PM_SaberLockResultAnim( gent, breakType, SABERLOCK_WIN ); pm->ps->weaponstate = WEAPON_FIRING; if ( genemy && genemy->client ) { loseAnim = PM_SaberLockResultAnim( genemy, breakType, SABERLOCK_LOSE ); genemy->client->ps.weaponstate = WEAPON_READY; } } if ( d_saberCombat->integer ) { Com_Printf( "%s won saber lock, anim = %s!\n", gent->NPC_type, animTable[winAnim].name ); Com_Printf( "%s lost saber lock, anim = %s!\n", genemy->NPC_type, animTable[loseAnim].name ); } pm->ps->saberLockTime = genemy->client->ps.saberLockTime = 0; pm->ps->saberLockEnemy = genemy->client->ps.saberLockEnemy = ENTITYNUM_NONE; pm->ps->saberMoveNext = LS_NONE; if ( genemy && genemy->client ) { genemy->client->ps.saberMoveNext = LS_NONE; } PM_AddEvent( EV_JUMP ); if ( result == LOCK_STALEMATE ) {//no-one won G_AddEvent( genemy, EV_JUMP, 0 ); } else { if ( pm->ps->clientNum ) {//an NPC pm->ps->saberEventFlags |= SEF_LOCK_WON;//tell the winner to press the advantage } //painDebounceTime will stop them from doing anything genemy->painDebounceTime = level.time + genemy->client->ps.torsoAnimTimer + 500; if ( Q_irand( 0, 1 ) ) { G_AddEvent( genemy, EV_PAIN, Q_irand( 0, 75 ) ); } else { if ( genemy->NPC ) { genemy->NPC->blockedSpeechDebounceTime = 0; } G_AddVoiceEvent( genemy, Q_irand( EV_PUSHED1, EV_PUSHED3 ), 500 ); } if ( result == LOCK_VICTORY ) {//one person won if ( Q_irand( FORCE_LEVEL_1, FORCE_LEVEL_2 ) < pm->ps->forcePowerLevel[FP_SABER_OFFENSE] ) { vec3_t throwDir = {0,0,350}; int winMove = pm->ps->saberMove; if ( !singleVsSingle ) {//all others have their own super breaks //so it doesn't try to set some other anim below winAnim = -1; } else if ( winAnim == BOTH_LK_S_S_S_SB_1_W || winAnim == BOTH_LK_S_S_T_SB_1_W ) {//doing a superbreak on single-vs-single, don't do the old superbreaks this time //so it doesn't try to set some other anim below winAnim = -1; } else {//JK2-style switch ( winAnim ) { case BOTH_A3_T__B_: winAnim = BOTH_D1_TL___; winMove = LS_D1_TL; //FIXME: mod throwDir? break; case BOTH_K1_S1_T_: //FIXME: mod throwDir? break; case BOTH_CWCIRCLEBREAK: //FIXME: mod throwDir? break; case BOTH_CCWCIRCLEBREAK: winAnim = BOTH_A1_BR_TL; winMove = LS_A_BR2TL; //FIXME: mod throwDir? break; } if ( winAnim != BOTH_CCWCIRCLEBREAK ) { if ( (!genemy->s.number&&genemy->health<=25)//player low on health ||(genemy->s.number&&genemy->client->NPC_class!=CLASS_KYLE&&genemy->client->NPC_class!=CLASS_LUKE&&genemy->client->NPC_class!=CLASS_TAVION&&genemy->client->NPC_class!=CLASS_ALORA&&genemy->client->NPC_class!=CLASS_DESANN)//any NPC that's not a boss character ||(genemy->s.number&&genemy->health<=50) )//boss character with less than 50 health left {//possibly knock saber out of hand OR cut hand off! if ( Q_irand( 0, 25 ) < victoryStrength && ((!genemy->s.number&&genemy->health<=10)||genemy->s.number) ) { NPC_SetAnim( genemy, SETANIM_BOTH, BOTH_RIGHTHANDCHOPPEDOFF, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD );//force this genemy->client->dismembered = false; G_DoDismemberment( genemy, genemy->client->renderInfo.handRPoint, MOD_SABER, 1000, HL_HAND_RT, qtrue ); G_Damage( genemy, gent, gent, throwDir, genemy->client->renderInfo.handRPoint, genemy->health+10, DAMAGE_NO_PROTECTION|DAMAGE_NO_ARMOR|DAMAGE_NO_KNOCKBACK|DAMAGE_NO_HIT_LOC, MOD_SABER, HL_NONE ); PM_SetAnim( pm, SETANIM_BOTH, winAnim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); pm->ps->weaponTime = pm->ps->torsoAnimTimer + 500; pm->ps->saberMove = winMove; pm->ps->saberBlocked = BLOCKED_NONE; pm->ps->weaponstate = WEAPON_FIRING; return; } } } } //else see if we can knock the saber out of their hand //FIXME: for now, always disarm the right-hand saber if ( !(genemy->client->ps.saber[0].saberFlags&SFL_NOT_DISARMABLE) ) { //add disarmBonus into this check victoryStrength += pm->ps->SaberDisarmBonus( 0 )*2; if ( (genemy->client->ps.saber[0].saberFlags&SFL_TWO_HANDED) || (genemy->client->ps.dualSabers && genemy->client->ps.saber[1].Active()) ) {//defender gets a bonus for using a 2-handed saber or 2 sabers victoryStrength -= 2; } if ( pm->ps->forcePowersActive&(1<client->ps.forcePowerLevel[FP_RAGE]; } else if ( pm->ps->forceRageRecoveryTime > pm->cmd.serverTime ) { victoryStrength--; } if ( genemy->client->ps.forceRageRecoveryTime > pm->cmd.serverTime ) { victoryStrength++; } if ( Q_irand( 0, 10 ) < victoryStrength ) { if ( !(genemy->client->ps.saber[0].saberFlags&SFL_TWO_HANDED) || !Q_irand( 0, 1 ) ) {//if it's a two-handed saber, it has a 50% chance of resisting a disarming WP_SaberLose( genemy, throwDir ); if ( winAnim != -1 ) { PM_SetAnim( pm, SETANIM_BOTH, winAnim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); pm->ps->weaponTime = pm->ps->torsoAnimTimer; pm->ps->saberMove = winMove; pm->ps->saberBlocked = BLOCKED_NONE; pm->ps->weaponstate = WEAPON_FIRING; } } } } } } } } int G_SaberLockStrength( gentity_t *gent ) { int strength = gent->client->ps.saber[0].lockBonus; if ( (gent->client->ps.saber[0].saberFlags&SFL_TWO_HANDED) ) { strength += 1; } if ( gent->client->ps.dualSabers && gent->client->ps.saber[1].Active() ) { strength += 1 + gent->client->ps.saber[1].lockBonus; } if ( gent->client->ps.forcePowersActive&(1<client->ps.forcePowerLevel[FP_RAGE]; } else if ( gent->client->ps.forceRageRecoveryTime > pm->cmd.serverTime ) { strength--; } if ( gent->s.number >= MAX_CLIENTS ) { if ( gent->client->NPC_class == CLASS_DESANN || gent->client->NPC_class == CLASS_LUKE ) { strength += 5+Q_irand(0,g_spskill->integer); } else { strength += gent->client->ps.forcePowerLevel[FP_SABER_OFFENSE]+Q_irand(0,g_spskill->integer); if ( gent->NPC ) { if ( (gent->NPC->aiFlags&NPCAI_BOSS_CHARACTER) || (gent->NPC->aiFlags&NPCAI_ROSH) || gent->client->NPC_class == CLASS_SHADOWTROOPER ) { strength += Q_irand(0,2); } else if ( (gent->NPC->aiFlags&NPCAI_SUBBOSS_CHARACTER) ) { strength += Q_irand(-1,1); } } } } else {//player strength += gent->client->ps.forcePowerLevel[FP_SABER_OFFENSE]+Q_irand(0,g_spskill->integer)+Q_irand(0,1); } return strength; } qboolean PM_InSaberLockOld( int anim ) { switch ( anim ) { case BOTH_BF2LOCK: case BOTH_BF1LOCK: case BOTH_CWCIRCLELOCK: case BOTH_CCWCIRCLELOCK: return qtrue; } return qfalse; } qboolean PM_InSaberLock( int anim ) { switch ( anim ) { case BOTH_LK_S_DL_S_L_1: //lock if I'm using single vs. a dual case BOTH_LK_S_DL_T_L_1: //lock if I'm using single vs. a dual case BOTH_LK_S_ST_S_L_1: //lock if I'm using single vs. a staff case BOTH_LK_S_ST_T_L_1: //lock if I'm using single vs. a staff case BOTH_LK_S_S_S_L_1: //lock if I'm using single vs. a single and I initiated case BOTH_LK_S_S_T_L_1: //lock if I'm using single vs. a single and I initiated case BOTH_LK_DL_DL_S_L_1: //lock if I'm using dual vs. dual and I initiated case BOTH_LK_DL_DL_T_L_1: //lock if I'm using dual vs. dual and I initiated case BOTH_LK_DL_ST_S_L_1: //lock if I'm using dual vs. a staff case BOTH_LK_DL_ST_T_L_1: //lock if I'm using dual vs. a staff case BOTH_LK_DL_S_S_L_1: //lock if I'm using dual vs. a single case BOTH_LK_DL_S_T_L_1: //lock if I'm using dual vs. a single case BOTH_LK_ST_DL_S_L_1: //lock if I'm using staff vs. dual case BOTH_LK_ST_DL_T_L_1: //lock if I'm using staff vs. dual case BOTH_LK_ST_ST_S_L_1: //lock if I'm using staff vs. a staff and I initiated case BOTH_LK_ST_ST_T_L_1: //lock if I'm using staff vs. a staff and I initiated case BOTH_LK_ST_S_S_L_1: //lock if I'm using staff vs. a single case BOTH_LK_ST_S_T_L_1: //lock if I'm using staff vs. a single case BOTH_LK_S_S_S_L_2: case BOTH_LK_S_S_T_L_2: case BOTH_LK_DL_DL_S_L_2: case BOTH_LK_DL_DL_T_L_2: case BOTH_LK_ST_ST_S_L_2: case BOTH_LK_ST_ST_T_L_2: return qtrue; break; default: return PM_InSaberLockOld( anim ); break; } //return qfalse; } extern qboolean ValidAnimFileIndex ( int index ); extern qboolean G_CheckIncrementLockAnim( int anim, int winOrLose ); qboolean PM_SaberLocked( void ) { //FIXME: maybe kick out of saberlock? if ( pm->ps->saberLockEnemy == ENTITYNUM_NONE ) { if ( PM_InSaberLock( pm->ps->torsoAnim ) ) {//wtf? Maybe enemy died? PM_SaberLockWinAnim( LOCK_STALEMATE, SABERLOCK_BREAK ); } return qfalse; } gentity_t *gent = pm->gent; if ( !gent ) { return qfalse; } gentity_t *genemy = &g_entities[pm->ps->saberLockEnemy]; if ( !genemy ) { return qfalse; } if ( PM_InSaberLock( pm->ps->torsoAnim ) && PM_InSaberLock( genemy->client->ps.torsoAnim ) ) { if ( pm->ps->saberLockTime <= level.time + 500 && pm->ps->saberLockEnemy != ENTITYNUM_NONE ) {//lock just ended int strength = G_SaberLockStrength( gent ); int eStrength = G_SaberLockStrength( genemy ); if ( strength > 1 && eStrength > 1 && !Q_irand( 0, abs(strength-eStrength)+1 ) ) {//both knock each other down! PM_SaberLockBreak( gent, genemy, LOCK_DRAW, 0 ); } else {//both "win" PM_SaberLockBreak( gent, genemy, LOCK_STALEMATE, 0 ); } return qtrue; } else if ( pm->ps->saberLockTime < level.time ) {//done... tie breaker above should have handled this, but...? if ( PM_InSaberLock( pm->ps->torsoAnim ) && pm->ps->torsoAnimTimer > 0 ) { pm->ps->torsoAnimTimer = 0; } if ( PM_InSaberLock( pm->ps->legsAnim ) && pm->ps->legsAnimTimer > 0 ) { pm->ps->legsAnimTimer = 0; } return qfalse; } else if ( pm->cmd.buttons & BUTTON_ATTACK ) {//holding attack if ( !(pm->ps->pm_flags&PMF_ATTACK_HELD) ) {//tapping int remaining = 0; if( ValidAnimFileIndex( gent->client->clientInfo.animFileIndex ) ) { animation_t *anim; float currentFrame, junk2; int curFrame, junk; int strength = 1; anim = &level.knownAnimFileSets[gent->client->clientInfo.animFileIndex].animations[pm->ps->torsoAnim]; #ifdef _DEBUG qboolean ret = #endif gi.G2API_GetBoneAnimIndex( &gent->ghoul2[gent->playerModel], gent->lowerLumbarBone, (cg.time?cg.time:level.time), ¤tFrame, &junk, &junk, &junk, &junk2, NULL ); #ifdef _DEBUG assert( ret ); // this would be pretty bad, the below code seems to assume the call succeeds. -gil #endif strength = G_SaberLockStrength( gent ); if ( PM_InSaberLockOld( pm->ps->torsoAnim ) ) {//old locks if ( pm->ps->torsoAnim == BOTH_CCWCIRCLELOCK || pm->ps->torsoAnim == BOTH_BF2LOCK ) { curFrame = floor( currentFrame )-strength; //drop my frame one if ( curFrame <= anim->firstFrame ) {//I won! Break out PM_SaberLockBreak( gent, genemy, LOCK_VICTORY, strength ); return qtrue; } else { PM_SetAnimFrame( gent, curFrame, qtrue, qtrue ); remaining = curFrame-anim->firstFrame; if ( d_saberCombat->integer ) { Com_Printf( "%s pushing in saber lock, %d frames to go!\n", gent->NPC_type, remaining ); } } } else { curFrame = ceil( currentFrame )+strength; //advance my frame one if ( curFrame >= anim->firstFrame+anim->numFrames ) {//I won! Break out PM_SaberLockBreak( gent, genemy, LOCK_VICTORY, strength ); return qtrue; } else { PM_SetAnimFrame( gent, curFrame, qtrue, qtrue ); remaining = anim->firstFrame+anim->numFrames-curFrame; if ( d_saberCombat->integer ) { Com_Printf( "%s pushing in saber lock, %d frames to go!\n", gent->NPC_type, remaining ); } } } } else {//new locks if ( G_CheckIncrementLockAnim( pm->ps->torsoAnim, SABERLOCK_WIN ) ) { curFrame = ceil( currentFrame )+strength; //advance my frame one if ( curFrame >= anim->firstFrame+anim->numFrames ) {//I won! Break out PM_SaberLockBreak( gent, genemy, LOCK_VICTORY, strength ); return qtrue; } else { PM_SetAnimFrame( gent, curFrame, qtrue, qtrue ); remaining = anim->firstFrame+anim->numFrames-curFrame; if ( d_saberCombat->integer ) { Com_Printf( "%s pushing in saber lock, %d frames to go!\n", gent->NPC_type, remaining ); } } } else { curFrame = floor( currentFrame )-strength; //drop my frame one if ( curFrame <= anim->firstFrame ) {//I won! Break out PM_SaberLockBreak( gent, genemy, LOCK_VICTORY, strength ); return qtrue; } else { PM_SetAnimFrame( gent, curFrame, qtrue, qtrue ); remaining = curFrame-anim->firstFrame; if ( d_saberCombat->integer ) { Com_Printf( "%s pushing in saber lock, %d frames to go!\n", gent->NPC_type, remaining ); } } } } if ( !Q_irand( 0, 2 ) ) { if ( pm->ps->clientNum < MAX_CLIENTS ) { if ( !Q_irand( 0, 3 ) ) { PM_AddEvent( EV_JUMP ); } else { PM_AddEvent( Q_irand( EV_PUSHED1, EV_PUSHED3 ) ); } } else { if ( gent->NPC && gent->NPC->blockedSpeechDebounceTime < level.time ) { switch ( Q_irand( 0, 3 ) ) { case 0: PM_AddEvent( EV_JUMP ); break; case 1: PM_AddEvent( Q_irand( EV_ANGER1, EV_ANGER3 ) ); gent->NPC->blockedSpeechDebounceTime = level.time + 3000; break; case 2: PM_AddEvent( Q_irand( EV_TAUNT1, EV_TAUNT3 ) ); gent->NPC->blockedSpeechDebounceTime = level.time + 3000; break; case 3: PM_AddEvent( Q_irand( EV_GLOAT1, EV_GLOAT3 ) ); gent->NPC->blockedSpeechDebounceTime = level.time + 3000; break; } } } } } else { return qfalse; } if( ValidAnimFileIndex( genemy->client->clientInfo.animFileIndex ) ) { animation_t *anim; anim = &level.knownAnimFileSets[genemy->client->clientInfo.animFileIndex].animations[genemy->client->ps.torsoAnim]; /* float currentFrame, junk2; int junk; gi.G2API_GetBoneAnimIndex( &genemy->ghoul2[genemy->playerModel], genemy->lowerLumbarBone, (cg.time?cg.time:level.time), ¤tFrame, &junk, &junk, &junk, &junk2, NULL ); */ if ( !Q_irand( 0, 2 ) ) { switch ( Q_irand( 0, 3 ) ) { case 0: G_AddEvent( genemy, EV_PAIN, floor((float)genemy->health/genemy->max_health*100.0f) ); break; case 1: G_AddVoiceEvent( genemy, Q_irand( EV_PUSHED1, EV_PUSHED3 ), 500 ); break; case 2: G_AddVoiceEvent( genemy, Q_irand( EV_CHOKE1, EV_CHOKE3 ), 500 ); break; case 3: G_AddVoiceEvent( genemy, EV_PUSHFAIL, 2000 ); break; } } if ( PM_InSaberLockOld( genemy->client->ps.torsoAnim ) ) { if ( genemy->client->ps.torsoAnim == BOTH_CCWCIRCLELOCK || genemy->client->ps.torsoAnim == BOTH_BF2LOCK ) { PM_SetAnimFrame( genemy, anim->firstFrame+anim->numFrames-remaining, qtrue, qtrue ); } else { PM_SetAnimFrame( genemy, anim->firstFrame+remaining, qtrue, qtrue ); } } else {//new locks //??? if ( G_CheckIncrementLockAnim( genemy->client->ps.torsoAnim, SABERLOCK_LOSE ) ) { PM_SetAnimFrame( genemy, anim->firstFrame+anim->numFrames-remaining, qtrue, qtrue ); } else { PM_SetAnimFrame( genemy, anim->firstFrame+remaining, qtrue, qtrue ); } } } } } else {//FIXME: other ways out of a saberlock? //force-push? (requires more force power?) //kick? (requires anim ... hit jump key?) //roll? //backflip? } } else {//something broke us out of it if ( gent->painDebounceTime > level.time && genemy->painDebounceTime > level.time ) { PM_SaberLockBreak( gent, genemy, LOCK_DRAW, 0 ); } else if ( gent->painDebounceTime > level.time ) { PM_SaberLockBreak( genemy, gent, LOCK_VICTORY, 0 ); } else if ( genemy->painDebounceTime > level.time ) { PM_SaberLockBreak( gent, genemy, LOCK_VICTORY, 0 ); } else { PM_SaberLockBreak( gent, genemy, LOCK_STALEMATE, 0 ); } } return qtrue; } qboolean G_EnemyInKickRange( gentity_t *self, gentity_t *enemy ) { if ( !self || !enemy ) { return qfalse; } if ( fabs(self->currentOrigin[2]-enemy->currentOrigin[2]) < 32 ) {//generally at same height if ( DistanceHorizontal( self->currentOrigin, enemy->currentOrigin ) <= (STAFF_KICK_RANGE+8.0f+(self->maxs[0]*1.5f)+(enemy->maxs[0]*1.5f)) ) {//within kicking range! return qtrue; } } return qfalse; } qboolean G_CanKickEntity( gentity_t *self, gentity_t *target ) { if ( target && target->client && !PM_InKnockDown( &target->client->ps ) && G_EnemyInKickRange( self, target ) ) { return qtrue; } return qfalse; } float PM_GroundDistance(void) { trace_t tr; vec3_t down; VectorCopy(pm->ps->origin, down); down[2] -= 4096; pm->trace(&tr, pm->ps->origin, pm->mins, pm->maxs, down, pm->ps->clientNum, pm->tracemask, (EG2_Collision)0, 0); VectorSubtract(pm->ps->origin, tr.endpos, down); return VectorLength(down); } float G_GroundDistance(gentity_t *self) { if ( !self ) {//wtf?!! return Q3_INFINITE; } trace_t tr; vec3_t down; VectorCopy(self->currentOrigin, down); down[2] -= 4096; gi.trace(&tr, self->currentOrigin, self->mins, self->maxs, down, self->s.number, self->clipmask, (EG2_Collision)0, 0); VectorSubtract(self->currentOrigin, tr.endpos, down); return VectorLength(down); } saberMoveName_t G_PickAutoKick( gentity_t *self, gentity_t *enemy, qboolean storeMove ) { saberMoveName_t kickMove = LS_NONE; if ( !self || !self->client ) { return LS_NONE; } if ( !enemy ) { return LS_NONE; } vec3_t v_fwd, v_rt, enemyDir, fwdAngs = {0,self->client->ps.viewangles[YAW],0}; VectorSubtract( enemy->currentOrigin, self->currentOrigin, enemyDir ); VectorNormalize( enemyDir );//not necessary, I guess, but doesn't happen often AngleVectors( fwdAngs, v_fwd, v_rt, NULL ); float fDot = DotProduct( enemyDir, v_fwd ); float rDot = DotProduct( enemyDir, v_rt ); if ( fabs( rDot ) > 0.5f && fabs( fDot ) < 0.5f ) {//generally to one side if ( rDot > 0 ) {//kick right kickMove = LS_KICK_R; } else {//kick left kickMove = LS_KICK_L; } } else if ( fabs( fDot ) > 0.5f && fabs( rDot ) < 0.5f ) {//generally in front or behind us if ( fDot > 0 ) {//kick fwd kickMove = LS_KICK_F; } else {//kick back kickMove = LS_KICK_B; } } else {//diagonal to us, kick would miss } if ( kickMove != LS_NONE ) {//have a valid one to do if ( self->client->ps.groundEntityNum == ENTITYNUM_NONE ) {//if in air, convert kick to an in-air kick float gDist = G_GroundDistance( self ); //let's only allow air kicks if a certain distance from the ground //it's silly to be able to do them right as you land. //also looks wrong to transition from a non-complete flip anim... if ((!PM_FlippingAnim( self->client->ps.legsAnim ) || self->client->ps.legsAnimTimer <= 0) && gDist > 64.0f && //strict minimum gDist > (-self->client->ps.velocity[2])-64.0f //make sure we are high to ground relative to downward velocity as well ) { switch ( kickMove ) { case LS_KICK_F: kickMove = LS_KICK_F_AIR; break; case LS_KICK_B: kickMove = LS_KICK_B_AIR; break; case LS_KICK_R: kickMove = LS_KICK_R_AIR; break; case LS_KICK_L: kickMove = LS_KICK_L_AIR; break; default: //oh well, can't do any other kick move while in-air kickMove = LS_NONE; break; } } else {//leave it as a normal kick unless we're too high up if ( gDist > 128.0f || self->client->ps.velocity[2] >= 0 ) { //off ground, but too close to ground kickMove = LS_NONE; } } } if ( storeMove ) { self->client->ps.saberMoveNext = kickMove; } } return kickMove; } saberMoveName_t PM_PickAutoKick( gentity_t *enemy ) { return G_PickAutoKick( pm->gent, enemy, qfalse ); } saberMoveName_t G_PickAutoMultiKick( gentity_t *self, qboolean allowSingles, qboolean storeMove ) { gentity_t *ent; gentity_t *entityList[MAX_GENTITIES]; vec3_t mins, maxs; int i, e; int radius = ((self->maxs[0]*1.5f)+(self->maxs[0]*1.5f)+STAFF_KICK_RANGE+24.0f);//a little wide on purpose vec3_t center; saberMoveName_t kickMove, bestKick = LS_NONE; float distToEnt, bestDistToEnt = Q3_INFINITE; gentity_t *bestEnt = NULL; int enemiesFront = 0; int enemiesBack = 0; int enemiesRight = 0; int enemiesLeft = 0; int enemiesSpin = 0; if ( !self || !self->client ) { return LS_NONE; } VectorCopy( self->currentOrigin, center ); for ( i = 0 ; i < 3 ; i++ ) { mins[i] = center[i] - radius; maxs[i] = center[i] + radius; } int numListedEntities = gi.EntitiesInBox( mins, maxs, entityList, MAX_GENTITIES ); for ( e = 0 ; e < numListedEntities ; e++ ) { ent = entityList[ e ]; if (ent == self) continue; if (ent->owner == self) continue; if ( !(ent->inuse) ) continue; //not a client? if ( !ent->client ) continue; //ally? if ( ent->client->playerTeam == self->client->playerTeam ) continue; //dead? if ( ent->health <= 0 ) continue; //too far? distToEnt = DistanceSquared( ent->currentOrigin, center ); if ( distToEnt > (radius*radius) ) continue; kickMove = G_PickAutoKick( self, ent, qfalse ); if ( kickMove == LS_KICK_F_AIR && kickMove == LS_KICK_B_AIR && kickMove == LS_KICK_R_AIR && kickMove == LS_KICK_L_AIR ) {//in air? Can't do multikicks } else { switch ( kickMove ) { case LS_KICK_F: enemiesFront++; break; case LS_KICK_B: enemiesBack++; break; case LS_KICK_R: enemiesRight++; break; case LS_KICK_L: enemiesLeft++; break; default: enemiesSpin++; break; } } if ( allowSingles ) { if ( kickMove != LS_NONE && distToEnt < bestDistToEnt ) { bestKick = kickMove; bestEnt = ent; } } } kickMove = LS_NONE; if ( self->client->ps.groundEntityNum != ENTITYNUM_NONE ) {//can't do the multikicks in air if ( enemiesFront && enemiesBack && (enemiesFront+enemiesBack)-(enemiesRight+enemiesLeft)>1 ) {//more enemies in front/back than left/right kickMove = LS_KICK_BF; } else if ( enemiesRight && enemiesLeft && (enemiesRight+enemiesLeft)-(enemiesFront+enemiesBack)>1 ) {//more enemies on left & right than front/back kickMove = LS_KICK_RL; } else if ( (enemiesFront || enemiesBack) && (enemiesRight || enemiesLeft) ) {//at least 2 enemies around us, not aligned kickMove = LS_KICK_S; } else if ( enemiesSpin > 1 ) {//at least 2 enemies around us, not aligned kickMove = LS_KICK_S; } } if ( kickMove == LS_NONE && bestKick != LS_NONE ) {//no good multi-kick move, but we do have a nice single-kick we found kickMove = bestKick; //get mad at him so he knows he's being targetted if ( (self->s.number < MAX_CLIENTS||G_ControlledByPlayer(self)) && bestEnt != NULL ) {//player G_SetEnemy( self, bestEnt ); } } if ( kickMove != LS_NONE ) { if ( storeMove ) { self->client->ps.saberMoveNext = kickMove; } } return kickMove; } qboolean PM_PickAutoMultiKick( qboolean allowSingles ) { saberMoveName_t kickMove = G_PickAutoMultiKick( pm->gent, allowSingles, qfalse ); if ( kickMove != LS_NONE ) { PM_SetSaberMove( kickMove ); return qtrue; } return qfalse; } qboolean PM_SaberThrowable( void ) { //ugh, hard-coding this is bad... if ( pm->ps->saberAnimLevel == SS_STAFF ) { return qfalse; } if ( !(pm->ps->saber[0].saberFlags&SFL_NOT_THROWABLE) ) {//yes, this saber is always throwable return qtrue; } //saber is not normally throwable if ( (pm->ps->saber[0].saberFlags&SFL_SINGLE_BLADE_THROWABLE) ) {//it is throwable if only one blade is on if ( pm->ps->saber[0].numBlades > 1 ) {//it has more than one blade int numBladesActive = 0; for ( int i = 0; i < pm->ps->saber[0].numBlades; i++ ) { if ( pm->ps->saber[0].blade[i].active ) { numBladesActive++; } } if ( numBladesActive == 1 ) {//only 1 blade is on return qtrue; } } } //nope, can't throw it return qfalse; } qboolean PM_CheckAltKickAttack( void ) { if ( (pm->cmd.buttons&BUTTON_ALT_ATTACK) && (!(pm->ps->pm_flags&PMF_ALT_ATTACK_HELD) ||PM_SaberInReturn(pm->ps->saberMove)) && (!PM_FlippingAnim(pm->ps->legsAnim)||pm->ps->legsAnimTimer<=250) && (!PM_SaberThrowable()) && pm->ps->SaberActive() && !(pm->ps->saber[0].saberFlags&SFL_NO_KICKS)//okay to do kicks with this saber && (!pm->ps->dualSabers || !(pm->ps->saber[1].saberFlags&SFL_NO_KICKS) )//okay to do kicks with this saber ) { return qtrue; } return qfalse; } qboolean PM_CheckUpsideDownAttack( void ) { if ( pm->ps->saberMove != LS_READY ) { return qfalse; } if ( !(pm->cmd.buttons&BUTTON_ATTACK) ) { return qfalse; } if ( pm->ps->saberAnimLevel < SS_FAST || pm->ps->saberAnimLevel > SS_STRONG ) { return qfalse; } if ( (pm->ps->clientNum >= MAX_CLIENTS&&!PM_ControlledByPlayer()) ) {//FIXME: check ranks? return qfalse; } //FIXME: enemy below //FIXME: more than 64 off ground if ( !g_debugMelee->integer ) {//hmm, can't get this to work quite the way we wanted... secret move, then! return qfalse; } switch( pm->ps->legsAnim ) { case BOTH_WALL_RUN_RIGHT_FLIP: case BOTH_WALL_RUN_LEFT_FLIP: case BOTH_WALL_FLIP_RIGHT: case BOTH_WALL_FLIP_LEFT: case BOTH_FLIP_BACK1: case BOTH_FLIP_BACK2: case BOTH_FLIP_BACK3: case BOTH_WALL_FLIP_BACK1: case BOTH_ALORA_FLIP_B: //JKA case BOTH_FORCEWALLRUNFLIP_END: { float animLength = PM_AnimLength( pm->gent->client->clientInfo.animFileIndex, (animNumber_t)pm->ps->legsAnim ); float elapsedTime = (float)(animLength-pm->ps->legsAnimTimer); float midPoint = animLength/2.0f; if ( elapsedTime < midPoint-100.0f || elapsedTime > midPoint+100.0f ) {//only a 200ms window (in middle of anim) of opportunity to do this move in these anims return qfalse; } } //NOTE: falls through on purpose case BOTH_FLIP_HOLD7: pm->ps->pm_flags |= PMF_SLOW_MO_FALL; PM_SetSaberMove( LS_UPSIDE_DOWN_ATTACK ); return qtrue; break; } return qfalse; } qboolean PM_SaberMoveOkayForKata( void ) { if ( g_saberNewControlScheme->integer ) { if ( pm->ps->saberMove == LS_READY //not doing anything || PM_SaberInReflect( pm->ps->saberMove ) )//interrupt a projectile blocking move { return qtrue; } else { return qfalse; } } else {//old control scheme, allow it to interrupt a start or ready if ( pm->ps->saberMove == LS_READY || PM_SaberInReflect( pm->ps->saberMove )//interrupt a projectile blocking move || PM_SaberInStart( pm->ps->saberMove ) ) { return qtrue; } else { return qfalse; } } } qboolean PM_CanDoKata( void ) { if ( PM_InSecondaryStyle() ) { return qfalse; } if ( !pm->ps->saberInFlight//not throwing saber && PM_SaberMoveOkayForKata() /* && pm->ps->saberAnimLevel >= SS_FAST//fast, med or strong style && pm->ps->saberAnimLevel <= SS_STRONG//FIXME: Tavion, too? */ && pm->ps->groundEntityNum != ENTITYNUM_NONE//not in the air && (pm->cmd.buttons&BUTTON_ATTACK)//pressing attack && pm->cmd.forwardmove >=0 //not moving back (used to be !pm->cmd.forwardmove) && !pm->cmd.rightmove//not moving r/l && pm->cmd.upmove <= 0//not jumping...? && G_TryingKataAttack(pm->gent,&pm->cmd)/*(pm->cmd.buttons&BUTTON_FORCE_FOCUS)*///holding focus && G_EnoughPowerForSpecialMove( pm->ps->forcePower, SABER_ALT_ATTACK_POWER, qtrue )/*pm->ps->forcePower >= SABER_ALT_ATTACK_POWER*/ )//SINGLE_SPECIAL_POWER )// have enough power {//FIXME: check rage, etc... return qtrue; } return qfalse; } void PM_SaberDroidWeapon( void ) { // make weapon function if ( pm->ps->weaponTime > 0 ) { pm->ps->weaponTime -= pml.msec; if ( pm->ps->weaponTime <= 0 ) { pm->ps->weaponTime = 0; } } // Now we react to a block action by the player's lightsaber. if ( pm->ps->saberBlocked ) { switch ( pm->ps->saberBlocked ) { case BLOCKED_PARRY_BROKEN: PM_SetAnim( pm, SETANIM_BOTH, Q_irand(BOTH_PAIN1,BOTH_PAIN3), SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); pm->ps->weaponTime = pm->ps->legsAnimTimer; break; case BLOCKED_ATK_BOUNCE: PM_SetAnim( pm, SETANIM_BOTH, Q_irand(BOTH_PAIN1,BOTH_PAIN3), SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); pm->ps->weaponTime = pm->ps->legsAnimTimer; break; case BLOCKED_UPPER_RIGHT: case BLOCKED_UPPER_RIGHT_PROJ: case BLOCKED_LOWER_RIGHT: case BLOCKED_LOWER_RIGHT_PROJ: PM_SetAnim( pm, SETANIM_BOTH, BOTH_P1_S1_TR, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); pm->ps->legsAnimTimer += Q_irand( 200, 1000 ); pm->ps->weaponTime = pm->ps->legsAnimTimer; break; case BLOCKED_UPPER_LEFT: case BLOCKED_UPPER_LEFT_PROJ: case BLOCKED_LOWER_LEFT: case BLOCKED_LOWER_LEFT_PROJ: PM_SetAnim( pm, SETANIM_BOTH, BOTH_P1_S1_TL, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); pm->ps->legsAnimTimer += Q_irand( 200, 1000 ); pm->ps->weaponTime = pm->ps->legsAnimTimer; break; case BLOCKED_TOP: case BLOCKED_TOP_PROJ: PM_SetAnim( pm, SETANIM_BOTH, BOTH_P1_S1_T_, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); pm->ps->legsAnimTimer += Q_irand( 200, 1000 ); pm->ps->weaponTime = pm->ps->legsAnimTimer; break; default: pm->ps->saberBlocked = BLOCKED_NONE; break; } pm->ps->saberBlocked = BLOCKED_NONE; pm->ps->saberBounceMove = LS_NONE; pm->ps->weaponstate = WEAPON_READY; // Done with block, so stop these active weapon branches. return; } } void PM_TryGrab( void ) { if ( pm->ps->groundEntityNum != ENTITYNUM_NONE //&& !pm->ps->saberInFlight && pm->ps->weaponTime <= 0 )//< 200 ) { PM_SetAnim( pm, SETANIM_BOTH, BOTH_KYLE_GRAB, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); pm->ps->torsoAnimTimer += 200; pm->ps->weaponTime = pm->ps->torsoAnimTimer; pm->ps->saberMove = pm->ps->saberMoveNext = LS_READY; VectorClear( pm->ps->velocity ); VectorClear( pm->ps->moveDir ); pm->cmd.rightmove = pm->cmd.forwardmove = pm->cmd.upmove = 0; if ( pm->gent ) { pm->gent->painDebounceTime = level.time + pm->ps->torsoAnimTimer; } pm->ps->SaberDeactivate(); } } void PM_TryAirKick( saberMoveName_t kickMove ) { if ( pm->ps->groundEntityNum < ENTITYNUM_NONE ) {//just do it PM_SetSaberMove( kickMove ); } else { float gDist = PM_GroundDistance(); //let's only allow air kicks if a certain distance from the ground //it's silly to be able to do them right as you land. //also looks wrong to transition from a non-complete flip anim... if ((!PM_FlippingAnim( pm->ps->legsAnim ) || pm->ps->legsAnimTimer <= 0) && gDist > 64.0f && //strict minimum gDist > (-pm->ps->velocity[2])-64.0f //make sure we are high to ground relative to downward velocity as well ) { PM_SetSaberMove( kickMove ); } else {//leave it as a normal kick unless we're too high up if ( gDist > 128.0f || pm->ps->velocity[2] >= 0 ) { //off ground, but too close to ground } else {//high close enough to ground to do a normal kick, convert it switch ( kickMove ) { case LS_KICK_F_AIR: PM_SetSaberMove( LS_KICK_F ); break; case LS_KICK_B_AIR: PM_SetSaberMove( LS_KICK_B ); break; case LS_KICK_R_AIR: PM_SetSaberMove( LS_KICK_R ); break; case LS_KICK_L_AIR: PM_SetSaberMove( LS_KICK_L ); break; default: break; } } } } } void PM_CheckKick( void ) { if ( !PM_KickMove( pm->ps->saberMove )//not already in a kick && !(pm->ps->pm_flags&PMF_DUCKED)//not ducked && (pm->cmd.upmove >= 0 ) )//not trying to duck {//player kicks //FIXME: only if FP_SABER_OFFENSE >= 3 if ( pm->cmd.rightmove ) {//kick to side if ( pm->cmd.rightmove > 0 ) {//kick right if ( pm->ps->groundEntityNum == ENTITYNUM_NONE || pm->cmd.upmove > 0 ) { PM_TryAirKick( LS_KICK_R_AIR ); } else { PM_SetSaberMove( LS_KICK_R ); } } else {//kick left if ( pm->ps->groundEntityNum == ENTITYNUM_NONE || pm->cmd.upmove > 0 ) { PM_TryAirKick( LS_KICK_L_AIR ); } else { PM_SetSaberMove( LS_KICK_L ); } } pm->cmd.rightmove = 0; } else if ( pm->cmd.forwardmove ) {//kick front/back if ( pm->cmd.forwardmove > 0 ) {//kick fwd if ( pm->ps->groundEntityNum == ENTITYNUM_NONE || pm->cmd.upmove > 0 ) { PM_TryAirKick( LS_KICK_F_AIR ); } /* else if ( pm->ps->weapon == WP_SABER && pm->ps->saberAnimLevel == SS_STAFF && pm->gent && G_CheckEnemyPresence( pm->gent, DIR_FRONT, 64, 0.8f ) ) {//FIXME: don't jump while doing this move and don't do this move if in air PM_SetSaberMove( LS_HILT_BASH ); } */ else { PM_SetSaberMove( LS_KICK_F ); } } else if ( pm->ps->groundEntityNum == ENTITYNUM_NONE || pm->cmd.upmove > 0 ) { PM_TryAirKick( LS_KICK_B_AIR ); } else {//kick back PM_SetSaberMove( LS_KICK_B ); } pm->cmd.forwardmove = 0; } else if ( pm->gent && pm->gent->enemy && G_CanKickEntity( pm->gent, pm->gent->enemy ) ) {//auto-pick? if ( /*(pm->ps->pm_flags&PMF_ALT_ATTACK_HELD) && (pm->cmd.buttons&BUTTON_ATTACK) &&*/ PM_PickAutoMultiKick( qfalse ) ) {//kicked! if ( pm->ps->saberMove == LS_KICK_RL ) {//just pull back if ( d_slowmodeath->integer > 3 ) { G_StartMatrixEffect( pm->gent, MEF_NO_SPIN, pm->ps->legsAnimTimer+500 ); } } else {//normal spin if ( d_slowmodeath->integer > 3 ) { G_StartMatrixEffect( pm->gent, 0, pm->ps->legsAnimTimer+500 ); } } if ( pm->ps->groundEntityNum == ENTITYNUM_NONE &&( pm->ps->saberMove == LS_KICK_S ||pm->ps->saberMove == LS_KICK_BF ||pm->ps->saberMove == LS_KICK_RL ) ) {//in the air and doing a jump-kick, which is a ground anim, so.... //cut z velocity...? pm->ps->velocity[2] = 0; } pm->cmd.upmove = 0; } else { saberMoveName_t kickMove = PM_PickAutoKick( pm->gent->enemy ); if ( kickMove != LS_NONE ) {//Matrix? PM_SetSaberMove( kickMove ); int meFlags = 0; switch ( kickMove ) { case LS_KICK_B://just pull back case LS_KICK_B_AIR://just pull back meFlags = MEF_NO_SPIN; break; case LS_KICK_L://spin to the left case LS_KICK_L_AIR://spin to the left meFlags = MEF_REVERSE_SPIN; break; default: break; } if ( d_slowmodeath->integer > 3 ) { G_StartMatrixEffect( pm->gent, meFlags, pm->ps->legsAnimTimer+500 ); } } } } else { if ( PM_PickAutoMultiKick( qtrue ) ) { int meFlags = 0; switch ( pm->ps->saberMove ) { case LS_KICK_RL://just pull back case LS_KICK_B://just pull back case LS_KICK_B_AIR://just pull back meFlags = MEF_NO_SPIN; break; case LS_KICK_L://spin to the left case LS_KICK_L_AIR://spin to the left meFlags = MEF_REVERSE_SPIN; break; default: break; } if ( d_slowmodeath->integer > 3 ) { G_StartMatrixEffect( pm->gent, meFlags, pm->ps->legsAnimTimer+500 ); } if ( pm->ps->groundEntityNum == ENTITYNUM_NONE &&( pm->ps->saberMove == LS_KICK_S ||pm->ps->saberMove == LS_KICK_BF ||pm->ps->saberMove == LS_KICK_RL ) ) {//in the air and doing a jump-kick, which is a ground anim, so.... //cut z velocity...? pm->ps->velocity[2] = 0; } pm->cmd.upmove = 0; } } } } void PM_CheckClearSaberBlock( void ) { if ( pm->ps->clientNum < MAX_CLIENTS || PM_ControlledByPlayer() ) {//player if ( pm->ps->saberBlocked >= BLOCKED_UPPER_RIGHT_PROJ && pm->ps->saberBlocked <= BLOCKED_TOP_PROJ ) {//blocking a projectile if ( pm->ps->forcePowerDebounce[FP_SABER_DEFENSE] < level.time ) {//block is done or breaking out of it with an attack pm->ps->weaponTime = 0; pm->ps->saberBlocked = BLOCKED_NONE; } else if ( (pm->cmd.buttons&BUTTON_ATTACK) ) {//block is done or breaking out of it with an attack pm->ps->weaponTime = 0; pm->ps->saberBlocked = BLOCKED_NONE; } } else if ( pm->ps->saberBlocked == BLOCKED_UPPER_LEFT && pm->ps->powerups[PW_SHOCKED] > level.time ) {//probably blocking lightning if ( (pm->cmd.buttons&BUTTON_ATTACK) ) {//trying to attack //allow the attack pm->ps->weaponTime = 0; pm->ps->saberBlocked = BLOCKED_NONE; } } } } qboolean PM_SaberBlocking( void ) { // Now we react to a block action by the player's lightsaber. if ( pm->ps->saberBlocked ) { if ( pm->ps->saberMove > LS_PUTAWAY && pm->ps->saberMove <= LS_A_BL2TR && pm->ps->saberBlocked != BLOCKED_PARRY_BROKEN && (pm->ps->saberBlocked < BLOCKED_UPPER_RIGHT_PROJ || pm->ps->saberBlocked > BLOCKED_TOP_PROJ))//&& Q_irand( 0, 2 ) {//we parried another lightsaber while attacking, so treat it as a bounce pm->ps->saberBlocked = BLOCKED_ATK_BOUNCE; } else if ( pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer() )//player { if ( pm->ps->saberBlocked >= BLOCKED_UPPER_RIGHT_PROJ && pm->ps->saberBlocked <= BLOCKED_TOP_PROJ ) {//blocking a projectile if ( (pm->cmd.buttons&BUTTON_ATTACK) ) {//trying to attack if ( pm->ps->saberMove == LS_READY || PM_SaberInReflect( pm->ps->saberMove ) ) {//not already busy or already in projectile deflection //trying to attack during projectile blocking, so do the attack instead pm->ps->saberBlocked = BLOCKED_NONE; pm->ps->saberBounceMove = LS_NONE; pm->ps->weaponstate = WEAPON_READY; if ( PM_SaberInReflect( pm->ps->saberMove ) && pm->ps->weaponTime > 0 ) {//interrupt the current deflection move pm->ps->weaponTime = 0; } return qfalse; } } } } /* else if ( (pm->cmd.buttons&BUTTON_ATTACK) && pm->ps->saberBlocked >= BLOCKED_UPPER_RIGHT && pm->ps->saberBlocked <= BLOCKED_TOP && !PM_SaberInKnockaway( pm->ps->saberBounceMove ) ) {//if hitting attack during a parry (not already a knockaway) if ( pm->ps->forcePowerLevel[FP_SABER_DEFENSE] > FORCE_LEVEL_2 ) {//have high saber defense, turn the parry into a knockaway? //FIXME: this could actually be bad for us...? Leaves us open pm->ps->saberBounceMove = PM_KnockawayForParry( pm->ps->saberBlocked ); } } */ if ( pm->ps->saberBlocked != BLOCKED_ATK_BOUNCE ) {//can't attack for twice whatever your skill level's parry debounce time is if ( pm->ps->clientNum == 0 || PM_ControlledByPlayer() ) {//player if ( pm->ps->forcePowerLevel[FP_SABER_DEFENSE] <= FORCE_LEVEL_1 ) { pm->ps->weaponTime = parryDebounce[pm->ps->forcePowerLevel[FP_SABER_DEFENSE]]; } } else {//NPC //pm->ps->weaponTime = parryDebounce[pm->ps->forcePowerLevel[FP_SABER_DEFENSE]] * 2; if ( pm->gent ) { pm->ps->weaponTime = Jedi_ReCalcParryTime( pm->gent, EVASION_PARRY ); } else {//WTF??? pm->ps->weaponTime = parryDebounce[pm->ps->forcePowerLevel[FP_SABER_DEFENSE]] * 2; } } } switch ( pm->ps->saberBlocked ) { case BLOCKED_PARRY_BROKEN: //whatever parry we were is in now broken, play the appropriate knocked-away anim { saberMoveName_t nextMove; if ( PM_SaberInBrokenParry( pm->ps->saberBounceMove ) ) {//already have one...? nextMove = (saberMoveName_t)pm->ps->saberBounceMove; } else { nextMove = PM_BrokenParryForParry( (saberMoveName_t)pm->ps->saberMove ); } if ( nextMove != LS_NONE ) { PM_SetSaberMove( nextMove ); pm->ps->weaponTime = pm->ps->torsoAnimTimer; } else {//Maybe in a knockaway? } //pm->ps->saberBounceMove = LS_NONE; } break; case BLOCKED_ATK_BOUNCE: // If there is absolutely no blocked move in the chart, don't even mess with the animation. // OR if we are already in a block or parry. if ( pm->ps->saberMove >= LS_T1_BR__R/*LS_BOUNCE_TOP*/ )//|| saberMoveData[pm->ps->saberMove].bounceMove == LS_NONE ) {//an actual bounce? Other bounces before this are actually transitions? pm->ps->saberBlocked = BLOCKED_NONE; } else if ( PM_SaberInBounce( pm->ps->saberMove ) || !PM_SaberInAttack( pm->ps->saberMove ) ) {//already in the bounce, go into an attack or transition to ready.. should never get here since can't be blocked in a bounce! int nextMove; if ( pm->cmd.buttons & BUTTON_ATTACK ) {//transition to a new attack if ( pm->ps->clientNum && !PM_ControlledByPlayer() ) {//NPC nextMove = saberMoveData[pm->ps->saberMove].chain_attack; } else {//player int newQuad = PM_SaberMoveQuadrantForMovement( &pm->cmd ); while ( newQuad == saberMoveData[pm->ps->saberMove].startQuad ) {//player is still in same attack quad, don't repeat that attack because it looks bad, //FIXME: try to pick one that might look cool? newQuad = Q_irand( Q_BR, Q_BL ); //FIXME: sanity check, just in case? }//else player is switching up anyway, take the new attack dir nextMove = transitionMove[saberMoveData[pm->ps->saberMove].startQuad][newQuad]; } } else {//return to ready if ( pm->ps->clientNum && !PM_ControlledByPlayer() ) {//NPC nextMove = saberMoveData[pm->ps->saberMove].chain_idle; } else {//player if ( saberMoveData[pm->ps->saberMove].startQuad == Q_T ) { nextMove = LS_R_BL2TR; } else if ( saberMoveData[pm->ps->saberMove].startQuad < Q_T ) { nextMove = LS_R_TL2BR+(saberMoveName_t)(saberMoveData[pm->ps->saberMove].startQuad-Q_BR); } else// if ( saberMoveData[pm->ps->saberMove].startQuad > Q_T ) { nextMove = LS_R_BR2TL+(saberMoveName_t)(saberMoveData[pm->ps->saberMove].startQuad-Q_TL); } } } PM_SetSaberMove( (saberMoveName_t)nextMove ); pm->ps->weaponTime = pm->ps->torsoAnimTimer; } else {//start the bounce move saberMoveName_t bounceMove; if ( pm->ps->saberBounceMove != LS_NONE ) { bounceMove = (saberMoveName_t)pm->ps->saberBounceMove; } else { bounceMove = PM_SaberBounceForAttack( (saberMoveName_t)pm->ps->saberMove ); } PM_SetSaberMove( bounceMove ); pm->ps->weaponTime = pm->ps->torsoAnimTimer; } //clear the saberBounceMove //pm->ps->saberBounceMove = LS_NONE; if (cg_debugSaber.integer>=2) { Com_Printf("Saber Block: Bounce\n"); } break; case BLOCKED_UPPER_RIGHT: if ( pm->ps->saberBounceMove != LS_NONE ) { PM_SetSaberMove( (saberMoveName_t)pm->ps->saberBounceMove ); //pm->ps->saberBounceMove = LS_NONE; pm->ps->weaponTime = pm->ps->torsoAnimTimer; } else { PM_SetSaberMove( LS_PARRY_UR ); } if ( cg_debugSaber.integer >= 2 ) { Com_Printf( "Saber Block: Parry UR\n" ); } break; case BLOCKED_UPPER_RIGHT_PROJ: PM_SetSaberMove( LS_REFLECT_UR ); if ( cg_debugSaber.integer >= 2 ) { Com_Printf("Saber Block: Deflect UR\n"); } break; case BLOCKED_UPPER_LEFT: if ( pm->ps->saberBounceMove != LS_NONE ) { PM_SetSaberMove( (saberMoveName_t)pm->ps->saberBounceMove ); //pm->ps->saberBounceMove = LS_NONE; pm->ps->weaponTime = pm->ps->torsoAnimTimer; } else { PM_SetSaberMove( LS_PARRY_UL ); } if ( cg_debugSaber.integer >= 2 ) { Com_Printf( "Saber Block: Parry UL\n" ); } break; case BLOCKED_UPPER_LEFT_PROJ: PM_SetSaberMove( LS_REFLECT_UL ); if ( cg_debugSaber.integer >= 2 ) { Com_Printf("Saber Block: Deflect UL\n"); } break; case BLOCKED_LOWER_RIGHT: if ( pm->ps->saberBounceMove != LS_NONE ) { PM_SetSaberMove( (saberMoveName_t)pm->ps->saberBounceMove ); //pm->ps->saberBounceMove = LS_NONE; pm->ps->weaponTime = pm->ps->torsoAnimTimer; } else { PM_SetSaberMove( LS_PARRY_LR ); } if ( cg_debugSaber.integer >= 2 ) { Com_Printf("Saber Block: Parry LR\n"); } break; case BLOCKED_LOWER_RIGHT_PROJ: PM_SetSaberMove( LS_REFLECT_LR ); if ( cg_debugSaber.integer >= 2 ) { Com_Printf("Saber Block: Deflect LR\n"); } break; case BLOCKED_LOWER_LEFT: if ( pm->ps->saberBounceMove != LS_NONE ) { PM_SetSaberMove( (saberMoveName_t)pm->ps->saberBounceMove ); //pm->ps->saberBounceMove = LS_NONE; pm->ps->weaponTime = pm->ps->torsoAnimTimer; } else { PM_SetSaberMove( LS_PARRY_LL ); } if ( cg_debugSaber.integer >= 2 ) { Com_Printf("Saber Block: Parry LL\n"); } break; case BLOCKED_LOWER_LEFT_PROJ: PM_SetSaberMove( LS_REFLECT_LL); if ( cg_debugSaber.integer >= 2 ) { Com_Printf("Saber Block: Deflect LL\n"); } break; case BLOCKED_TOP: if ( pm->ps->saberBounceMove != LS_NONE ) { PM_SetSaberMove( (saberMoveName_t)pm->ps->saberBounceMove ); //pm->ps->saberBounceMove = LS_NONE; pm->ps->weaponTime = pm->ps->torsoAnimTimer; } else { PM_SetSaberMove( LS_PARRY_UP ); } if ( cg_debugSaber.integer >= 2 ) { Com_Printf("Saber Block: Parry Top\n"); } break; case BLOCKED_TOP_PROJ: PM_SetSaberMove( LS_REFLECT_UP ); if ( cg_debugSaber.integer >= 2 ) { Com_Printf("Saber Block: Deflect Top\n"); } break; default: pm->ps->saberBlocked = BLOCKED_NONE; break; } // Charging is like a lead-up before attacking again. This is an appropriate use, or we can create a new weaponstate for blocking pm->ps->saberBounceMove = LS_NONE; pm->ps->weaponstate = WEAPON_READY; // Done with block, so stop these active weapon branches. return qtrue; } return qfalse; } qboolean PM_NPCCheckAttackRoll( void ) { if ( pm->ps->clientNum >= MAX_CLIENTS && !PM_ControlledByPlayer()//NPC && pm->gent && pm->gent->NPC //&& Q_irand(-3,pm->gent->NPC->rank)>RANK_CREWMAN) && ( pm->gent->NPC->rank > RANK_CREWMAN && !Q_irand(0,3-g_spskill->integer) ) && pm->gent->enemy && fabs(pm->gent->enemy->currentOrigin[2]-pm->ps->origin[2])<32 && DistanceHorizontalSquared(pm->gent->enemy->currentOrigin, pm->ps->origin ) < (128.0f*128.0f) && InFOV( pm->gent->enemy->currentOrigin, pm->ps->origin, pm->ps->viewangles, 30, 90 ) ) {//stab! return qtrue; } return qfalse; } /* ================= PM_WeaponLightsaber Consults a chart to choose what to do with the lightsaber. While this is a little different than the Quake 3 code, there is no clean way of using the Q3 code for this kind of thing. ================= */ // Ultimate goal is to set the sabermove to the proper next location // Note that if the resultant animation is NONE, then the animation is essentially "idle", and is set in WP_TorsoAnim void PM_WeaponLightsaber(void) { int addTime; qboolean delayed_fire = qfalse, animLevelOverridden = qfalse; int anim=-1; int curmove, newmove=LS_NONE; if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_SABER_DROID ) {//Saber droid does it's own attack logic PM_SaberDroidWeapon(); return; } // don't allow attack until all buttons are up if ( pm->ps->pm_flags & PMF_RESPAWNED ) { return; } // check for dead player if ( pm->ps->stats[STAT_HEALTH] <= 0 ) { if ( pm->gent ) { pm->gent->s.loopSound = 0; } return; } // make weapon function if ( pm->ps->weaponTime > 0 ) { pm->ps->weaponTime -= pml.msec; if ( pm->ps->weaponTime <= 0 ) { pm->ps->weaponTime = 0; } } if ( pm->ps->stats[STAT_WEAPONS]&(1<ps->dualSabers && pm->gent && pm->gent->weaponModel[1] ) {//holding scepter in left hand, use dual style pm->ps->saberAnimLevel = SS_DUAL; animLevelOverridden = qtrue; } else if ( pm->ps->saber[0].singleBladeStyle != SS_NONE//SaberStaff() && !pm->ps->dualSabers && pm->ps->saber[0].blade[0].active && !pm->ps->saber[0].blade[1].active ) {//using a staff, but only with first blade turned on, so use is as a normal saber...? //override so that single-blade staff must be used with specified style pm->ps->saberAnimLevel = pm->ps->saber[0].singleBladeStyle;//SS_STRONG; animLevelOverridden = qtrue; } else if ( pm->gent && cg.saberAnimLevelPending != pm->ps->saberAnimLevel && WP_SaberStyleValidForSaber( pm->gent, cg.saberAnimLevelPending ) ) {//go ahead and use the cg.saberAnimLevelPending below animLevelOverridden = qfalse; } else if ( pm->gent && ( WP_SaberStyleValidForSaber( pm->gent, pm->ps->saberAnimLevel ) || WP_UseFirstValidSaberStyle( pm->gent, &pm->ps->saberAnimLevel ) ) ) {//style we are using is not valid, switched us to a valid one animLevelOverridden = qtrue; } /* else if ( pm->ps->saber[0].Active() && pm->ps->saber[0].stylesAllowed ) {//one of the sabers I'm using forces me to use one of a set of styles if ( !(pm->ps->saber[0].stylesAllowed&(1<ps->saberAnimLevel)) ) {//I'm not currently using a valid one for ( int styleNum = SS_NONE+1; styleNum < SS_NUM_SABER_STYLES; styleNum++ ) {//loop through and use the first valid one if ( (pm->ps->saber[0].stylesAllowed&(1<ps->saberAnimLevel = styleNum; animLevelOverridden = qtrue; } } } } */ else if ( pm->ps->dualSabers ) { /* if ( pm->ps->saber[1].Active() && pm->ps->saber[1].stylesAllowed ) {//one of the sabers I'm using forces me to use one of a set of styles if ( !(pm->ps->saber[1].stylesAllowed&(1<ps->saberAnimLevel)) ) {//I'm not currently using a valid one for ( int styleNum = SS_NONE+1; styleNum < SS_NUM_SABER_STYLES; styleNum++ ) {//loop through and use the first valid one if ( (pm->ps->saber[1].stylesAllowed&(1<ps->saberAnimLevel = styleNum; animLevelOverridden = qtrue; } } } } else*/ if ( pm->ps->saber[1].Active() ) {//if second saber is on, must use dual style pm->ps->saberAnimLevel = SS_DUAL; animLevelOverridden = qtrue; } else if ( pm->ps->saber[0].Active() ) {//with only one saber on, use fast style pm->ps->saberAnimLevel = SS_FAST; animLevelOverridden = qtrue; } } if ( !animLevelOverridden ) { if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && cg.saberAnimLevelPending > SS_NONE && cg.saberAnimLevelPending != pm->ps->saberAnimLevel ) { if ( !PM_SaberInStart( pm->ps->saberMove ) && !PM_SaberInTransition( pm->ps->saberMove ) && !PM_SaberInAttack( pm->ps->saberMove ) ) {//don't allow changes when in the middle of an attack set...(or delay the change until it's done) pm->ps->saberAnimLevel = cg.saberAnimLevelPending; } } } else if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) ) {//if overrid the player's saberAnimLevel, let the cgame know cg.saberAnimLevelPending = pm->ps->saberAnimLevel; } /* if ( PM_InForceGetUp( pm->ps ) ) {//if mostly up, can start attack if ( pm->ps->torsoAnimTimer > 800 ) {//not up enough yet // make weapon function if ( pm->ps->weaponTime > 0 ) { pm->ps->weaponTime -= pml.msec; if ( pm->ps->weaponTime <= 0 ) { pm->ps->weaponTime = 0; } } return; } else { if ( pm->cmd.buttons & BUTTON_ATTACK ) {//let an attack interrupt the torso part of this force getup pm->ps->weaponTime = 0; } } } else */ if ( PM_InKnockDown( pm->ps ) || PM_InRoll( pm->ps )) {//in knockdown if ( pm->ps->legsAnim == BOTH_ROLL_F && pm->ps->legsAnimTimer <= 250 ) { if ( (pm->cmd.buttons&BUTTON_ATTACK) || PM_NPCCheckAttackRoll() ) { if ( G_EnoughPowerForSpecialMove( pm->ps->forcePower, SABER_ALT_ATTACK_POWER_FB ) ) { if ( !(pm->ps->saber[0].saberFlags&SFL_NO_ROLL_STAB) && (!pm->ps->dualSabers || !(pm->ps->saber[1].saberFlags&SFL_NO_ROLL_STAB)) ) {//okay to do roll-stab PM_SetSaberMove( LS_ROLL_STAB ); if ( pm->gent ) { G_DrainPowerForSpecialMove( pm->gent, FP_SABER_OFFENSE, SABER_ALT_ATTACK_POWER_FB ); } } } } } return; } if ( PM_SaberLocked() ) { pm->ps->saberMove = LS_NONE; return; } if ( pm->ps->torsoAnim == BOTH_FORCELONGLEAP_LAND || (pm->ps->torsoAnim == BOTH_FORCELONGLEAP_START && !(pm->cmd.buttons&BUTTON_ATTACK)) ) {//if you're in the long-jump and you're not attacking (or are landing), you're not doing anything if ( pm->ps->torsoAnimTimer ) { return; } } if ( pm->ps->legsAnim == BOTH_FLIP_HOLD7 && !(pm->cmd.buttons&BUTTON_ATTACK) ) {//if you're in the upside-down attack hold, don't do anything unless you're attacking return; } if ( PM_KickingAnim( pm->ps->legsAnim ) ) { if ( pm->ps->legsAnimTimer ) {//you're kicking, no interruptions return; } //done? be immeditately ready to do an attack pm->ps->saberMove = LS_READY; pm->ps->weaponTime = 0; } if ( pm->ps->saberMoveNext != LS_NONE && (pm->ps->saberMove == LS_READY||pm->ps->saberMove == LS_NONE))//ready for another one {//something is forcing us to set a specific next saberMove //FIXME: if this is a NPC kick, re-verify it before executing it! PM_SetSaberMove( (saberMoveName_t)pm->ps->saberMoveNext ); pm->ps->saberMoveNext = LS_NONE;//clear it now that we played it return; } if ( pm->ps->saberEventFlags&SEF_INWATER )//saber in water { pm->cmd.buttons &= ~(BUTTON_ATTACK|BUTTON_ALT_ATTACK|BUTTON_FORCE_FOCUS); } qboolean saberInAir = qtrue; if ( !PM_SaberInBrokenParry( pm->ps->saberMove ) && pm->ps->saberBlocked != BLOCKED_PARRY_BROKEN && !PM_DodgeAnim( pm->ps->torsoAnim ) && pm->ps->weaponstate != WEAPON_CHARGING_ALT && pm->ps->weaponstate != WEAPON_CHARGING) {//we're not stuck in a broken parry if ( pm->ps->saberInFlight ) {//guiding saber if ( pm->ps->saberEntityNum < ENTITYNUM_NONE && pm->ps->saberEntityNum > 0 )//player is 0 {// if ( &g_entities[pm->ps->saberEntityNum] != NULL && g_entities[pm->ps->saberEntityNum].s.pos.trType == TR_STATIONARY ) {//fell to the ground and we're not trying to pull it back saberInAir = qfalse; } } if ( pm->ps->weaponTime <= 0 || pm->ps->weaponstate != WEAPON_FIRING ) { if ( pm->ps->weapon != pm->cmd.weapon ) { PM_BeginWeaponChange( pm->cmd.weapon ); } } else if ( pm->ps->weapon == WP_SABER && (!pm->ps->dualSabers || !pm->ps->saber[1].Active()) ) {//guiding saber if ( saberInAir ) { if ( !PM_ForceAnim( pm->ps->torsoAnim ) || pm->ps->torsoAnimTimer < 300 ) {//don't interrupt a force power anim if ( pm->ps->torsoAnim != BOTH_LOSE_SABER || !pm->ps->torsoAnimTimer ) { PM_SetAnim( pm, SETANIM_TORSO, BOTH_SABERPULL, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); } } } return; } } } if ( pm->gent && pm->gent->client && pm->gent->client->fireDelay > 0 ) {//FIXME: this is going to fire off one frame before you expect, actually pm->gent->client->fireDelay -= pml.msec; if ( pm->gent->client->fireDelay <= 0 ) {//just finished delay timer pm->gent->client->fireDelay = 0; delayed_fire = qtrue; } } PM_CheckClearSaberBlock(); if ( PM_LockedAnim( pm->ps->torsoAnim ) && pm->ps->torsoAnimTimer ) {//can't interrupt these anims ever return; } if ( PM_SuperBreakLoseAnim( pm->ps->torsoAnim ) && pm->ps->torsoAnimTimer ) {//don't interrupt these anims return; } if ( PM_SuperBreakWinAnim( pm->ps->torsoAnim ) && pm->ps->torsoAnimTimer ) {//don't interrupt these anims return; } if ( PM_SaberBlocking() ) {//busy blocking, don't do attacks return; } // check for weapon change // can't change if weapon is firing, but can change again if lowering or raising if ( (pm->ps->weaponTime <= 0 || pm->ps->weaponstate != WEAPON_FIRING) && pm->ps->weaponstate != WEAPON_CHARGING_ALT && pm->ps->weaponstate != WEAPON_CHARGING) { if ( pm->ps->weapon != pm->cmd.weapon ) { PM_BeginWeaponChange( pm->cmd.weapon ); } } if ( pm->ps->weaponTime > 0 ) { //FIXME: allow some window of opportunity to change your attack // if it just started and your directional input is different // than it was before... but only 100 milliseconds at most? //OR: Make it so that attacks don't start until 100ms after you // press the attack button...??? if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) //player && PM_SaberInReturn( pm->ps->saberMove )//in a saber return move - FIXME: what about transitions? //&& pm->ps->torsoAnimTimer<=250//towards the end of a saber return anim && pm->ps->saberBlocked == BLOCKED_NONE//not interacting with any other saber && !(pm->cmd.buttons&BUTTON_ATTACK)//not trying to swing the saber && (pm->cmd.forwardmove||pm->cmd.rightmove) )//trying to kick in a specific direction { if ( PM_CheckAltKickAttack() )//trying to do a kick {//allow them to do the kick now! pm->ps->weaponTime = 0; PM_CheckKick(); return; } } else { if ( !pm->cmd.rightmove &&!pm->cmd.forwardmove &&(pm->cmd.buttons&BUTTON_ATTACK) ) { /* if ( PM_CheckDualSpinProtect() ) {//check to see if we're going to do the special dual push protect move PM_SetSaberMove( LS_DUAL_SPIN_PROTECT ); pm->ps->weaponstate = WEAPON_FIRING; return; } else */ if ( !g_saberNewControlScheme->integer ) { saberMoveName_t pullAtk = PM_CheckPullAttack(); if ( pullAtk != LS_NONE ) { PM_SetSaberMove( pullAtk ); pm->ps->weaponstate = WEAPON_FIRING; return; } } } else { return; } } } // ********************************************************* // WEAPON_DROPPING // ********************************************************* // change weapon if time if ( pm->ps->weaponstate == WEAPON_DROPPING ) { PM_FinishWeaponChange(); return; } // ********************************************************* // WEAPON_RAISING // ********************************************************* if ( pm->ps->weaponstate == WEAPON_RAISING ) {//Just selected the weapon pm->ps->weaponstate = WEAPON_IDLE; if(pm->gent && (pm->gent->s.numbergent))) { switch ( pm->ps->legsAnim ) { case BOTH_WALK1: case BOTH_WALK2: case BOTH_WALK_STAFF: case BOTH_WALK_DUAL: case BOTH_WALKBACK1: case BOTH_WALKBACK2: case BOTH_WALKBACK_STAFF: case BOTH_WALKBACK_DUAL: case BOTH_RUN1: case BOTH_RUN2: case BOTH_RUN_STAFF: case BOTH_RUN_DUAL: case BOTH_RUNBACK1: case BOTH_RUNBACK2: case BOTH_RUNBACK_STAFF: PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL); break; default: int anim = PM_ReadyPoseForSaberAnimLevel(); if (anim!=-1) { PM_SetAnim(pm,SETANIM_TORSO,anim,SETANIM_FLAG_NORMAL); } break; } } else { qboolean saberInAir = qtrue; if ( pm->ps->saberInFlight ) {//guiding saber if ( PM_SaberInBrokenParry( pm->ps->saberMove ) || pm->ps->saberBlocked == BLOCKED_PARRY_BROKEN || PM_DodgeAnim( pm->ps->torsoAnim ) ) {//we're stuck in a broken parry saberInAir = qfalse; } if ( pm->ps->saberEntityNum < ENTITYNUM_NONE && pm->ps->saberEntityNum > 0 )//player is 0 {// if ( &g_entities[pm->ps->saberEntityNum] != NULL && g_entities[pm->ps->saberEntityNum].s.pos.trType == TR_STATIONARY ) {//fell to the ground and we're not trying to pull it back saberInAir = qfalse; } } } if ( pm->ps->weapon == WP_SABER && pm->ps->saberInFlight && saberInAir && (!pm->ps->dualSabers || !pm->ps->saber[1].Active())) {//guiding saber if ( !PM_ForceAnim( pm->ps->torsoAnim ) || pm->ps->torsoAnimTimer < 300 ) {//don't interrupt a force power anim if ( pm->ps->torsoAnim != BOTH_LOSE_SABER || !pm->ps->torsoAnimTimer ) { PM_SetAnim( pm, SETANIM_TORSO, BOTH_SABERPULL, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); } } } else { // PM_SetAnim(pm, SETANIM_TORSO, BOTH_ATTACK1, SETANIM_FLAG_NORMAL); // Select the proper idle Lightsaber attack move from the chart. PM_SetSaberMove(LS_READY); } } return; } // ********************************************************* // Check for WEAPON ATTACK // ********************************************************* if ( PM_CanDoKata() ) { saberMoveName_t overrideMove = LS_INVALID; //see if we have an overridden (or cancelled) kata move if ( pm->ps->saber[0].kataMove != LS_INVALID ) { if ( pm->ps->saber[0].kataMove != LS_NONE ) { overrideMove = (saberMoveName_t)pm->ps->saber[0].kataMove; } } if ( overrideMove == LS_INVALID ) {//not overridden by first saber, check second if ( pm->ps->dualSabers ) { if ( pm->ps->saber[1].kataMove != LS_INVALID ) { if ( pm->ps->saber[1].kataMove != LS_NONE ) { overrideMove = (saberMoveName_t)pm->ps->saber[1].kataMove; } } } } //no overrides, cancelled? if ( overrideMove == LS_INVALID ) { if ( pm->ps->saber[0].kataMove == LS_NONE ) { overrideMove = LS_NONE; } else if ( pm->ps->dualSabers ) { if ( pm->ps->saber[1].kataMove == LS_NONE ) { overrideMove = LS_NONE; } } } if ( overrideMove == LS_INVALID ) {//not overridden //FIXME: make sure to turn on saber(s)! switch ( pm->ps->saberAnimLevel ) { case SS_FAST: case SS_TAVION: PM_SetSaberMove( LS_A1_SPECIAL ); break; case SS_MEDIUM: PM_SetSaberMove( LS_A2_SPECIAL ); break; case SS_STRONG: case SS_DESANN: PM_SetSaberMove( LS_A3_SPECIAL ); break; case SS_DUAL: PM_SetSaberMove( LS_DUAL_SPIN_PROTECT );//PM_CheckDualSpinProtect(); break; case SS_STAFF: PM_SetSaberMove( LS_STAFF_SOULCAL ); break; } pm->ps->weaponstate = WEAPON_FIRING; if ( pm->gent ) { G_DrainPowerForSpecialMove( pm->gent, FP_SABER_OFFENSE, SABER_ALT_ATTACK_POWER, qtrue );//FP_SPEED, SINGLE_SPECIAL_POWER ); //G_StartMatrixEffect( pm->gent, MEF_REVERSE_SPIN, pm->ps->torsoAnimTimer ); } } else if ( overrideMove != LS_NONE ) { PM_SetSaberMove( overrideMove ); pm->ps->weaponstate = WEAPON_FIRING; if ( pm->gent ) { G_DrainPowerForSpecialMove( pm->gent, FP_SABER_OFFENSE, SABER_ALT_ATTACK_POWER, qtrue );//FP_SPEED, SINGLE_SPECIAL_POWER ); //G_StartMatrixEffect( pm->gent, MEF_REVERSE_SPIN, pm->ps->torsoAnimTimer ); } } if ( overrideMove != LS_NONE ) {//not cancelled return; } } if ( PM_CheckAltKickAttack() ) {//trying to do a kick //FIXME: in-air kicks? if ( pm->ps->saberAnimLevel == SS_STAFF && (pm->ps->clientNum >= MAX_CLIENTS||PM_ControlledByPlayer()) ) {//NPCs spin the staff //NOTE: only NPCs can do it the easy way... they kick directly, not through ucmds... PM_SetSaberMove( LS_SPINATTACK ); return; } else { PM_CheckKick(); } return; } //this is never a valid regular saber attack button //pm->cmd.buttons &= ~BUTTON_FORCE_FOCUS; if ( PM_CheckUpsideDownAttack() ) { return; } if(!delayed_fire) { // Start with the current move, and cross index it with the current control states. if ( pm->ps->saberMove > LS_NONE && pm->ps->saberMove < LS_MOVE_MAX ) { curmove = (saberMoveName_t)pm->ps->saberMove; } else { curmove = LS_READY; } if ( curmove == LS_A_JUMP_T__B_ || pm->ps->torsoAnim == BOTH_FORCELEAP2_T__B_ ) {//must transition back to ready from this anim newmove = LS_R_T2B; } // check for fire else if ( !(pm->cmd.buttons & BUTTON_ATTACK) )//(BUTTON_ATTACK|BUTTON_ALT_ATTACK|BUTTON_FORCE_FOCUS)) ) {//not attacking pm->ps->weaponTime = 0; if ( pm->gent && pm->gent->client && pm->gent->client->fireDelay > 0 ) {//Still firing pm->ps->weaponstate = WEAPON_FIRING; } else if ( pm->ps->weaponstate != WEAPON_READY ) { pm->ps->weaponstate = WEAPON_IDLE; } //Check for finishing an anim if necc. if ( curmove >= LS_S_TL2BR && curmove <= LS_S_T2B ) {//started a swing, must continue from here newmove = LS_A_TL2BR + (curmove-LS_S_TL2BR); } else if ( curmove >= LS_A_TL2BR && curmove <= LS_A_T2B ) {//finished an attack, must continue from here newmove = LS_R_TL2BR + (curmove-LS_A_TL2BR); } else if ( PM_SaberInTransition( curmove ) ) {//in a transition, must play sequential attack newmove = saberMoveData[curmove].chain_attack; } else if ( PM_SaberInBounce( curmove ) ) {//in a bounce if ( pm->ps->clientNum && !PM_ControlledByPlayer() ) {//NPCs must play sequential attack //going into another attack... //allow endless chaining in level 1 attacks, several in level 2 and only one or a few in level 3 if ( PM_SaberKataDone( LS_NONE, LS_NONE ) ) {//done with this kata, must return to ready before attack again newmove = saberMoveData[curmove].chain_idle; } else {//okay to chain to another attack newmove = saberMoveData[curmove].chain_attack;//we assume they're attacking, even if they're not pm->ps->saberAttackChainCount++; } } else {//player gets his by directional control newmove = saberMoveData[curmove].chain_idle;//oops, not attacking, so don't chain } } else {//FIXME: what about returning from a parry? //PM_SetSaberMove( LS_READY ); if ( pm->ps->saberBlockingTime > cg.time ) { PM_SetSaberMove( LS_READY ); } return; } } // *************************************************** // Pressing attack, so we must look up the proper attack move. qboolean saberInAir = qtrue; if ( pm->ps->saberInFlight ) {//guiding saber if ( PM_SaberInBrokenParry( pm->ps->saberMove ) || pm->ps->saberBlocked == BLOCKED_PARRY_BROKEN || PM_DodgeAnim( pm->ps->torsoAnim ) ) {//we're stuck in a broken parry saberInAir = qfalse; } if ( pm->ps->saberEntityNum < ENTITYNUM_NONE && pm->ps->saberEntityNum > 0 )//player is 0 {// if ( &g_entities[pm->ps->saberEntityNum] != NULL && g_entities[pm->ps->saberEntityNum].s.pos.trType == TR_STATIONARY ) {//fell to the ground and we're not trying to pull it back saberInAir = qfalse; } } } if ( pm->ps->weapon == WP_SABER && pm->ps->saberInFlight && saberInAir && (!pm->ps->dualSabers || !pm->ps->saber[1].Active())) {//guiding saber if ( !PM_ForceAnim( pm->ps->torsoAnim ) || pm->ps->torsoAnimTimer < 300 ) {//don't interrupt a force power anim if ( pm->ps->torsoAnim != BOTH_LOSE_SABER || !pm->ps->torsoAnimTimer ) { PM_SetAnim( pm, SETANIM_TORSO,BOTH_SABERPULL,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); } } } else if ( pm->ps->weaponTime > 0 ) { // Last attack is not yet complete. pm->ps->weaponstate = WEAPON_FIRING; return; } else { int both = qfalse; if ( pm->ps->torsoAnim == BOTH_FORCELONGLEAP_ATTACK || pm->ps->torsoAnim == BOTH_FORCELONGLEAP_LAND ) {//can't attack in these anims return; } else if ( pm->ps->torsoAnim == BOTH_FORCELONGLEAP_START ) {//only 1 attack you can do from this anim if ( pm->ps->torsoAnimTimer >= 200 ) {//hit it early enough to do the attack PM_SetSaberMove( LS_LEAP_ATTACK ); } return; } if ( curmove >= LS_PARRY_UP && curmove <= LS_REFLECT_LL ) {//from a parry or reflection, can go directly into an attack if ( pm->ps->clientNum >= MAX_CLIENTS && !PM_ControlledByPlayer() ) {//NPCs newmove = PM_NPCSaberAttackFromQuad( saberMoveData[curmove].endQuad ); } else { newmove = PM_SaberAttackForMovement( pm->cmd.forwardmove, pm->cmd.rightmove, curmove ); } } if ( newmove != LS_NONE ) {//have a valid, final LS_ move picked, so skip findingt he transition move and just get the anim if (PM_HasAnimation( pm->gent, saberMoveData[newmove].animToUse)) { anim = saberMoveData[newmove].animToUse; } } //FIXME: diagonal dirs use the figure-eight attacks from ready pose? if ( anim == -1 ) { //FIXME: take FP_SABER_OFFENSE into account here somehow? if ( PM_SaberInTransition( curmove ) ) {//in a transition, must play sequential attack newmove = saberMoveData[curmove].chain_attack; } else if ( curmove >= LS_S_TL2BR && curmove <= LS_S_T2B ) {//started a swing, must continue from here newmove = LS_A_TL2BR + (curmove-LS_S_TL2BR); } else if ( PM_SaberInBrokenParry( curmove ) ) {//broken parries must always return to ready newmove = LS_READY; } else//if ( pm->cmd.buttons&BUTTON_ATTACK && !(pm->ps->pm_flags&PMF_ATTACK_HELD) )//only do this if just pressed attack button? {//get attack move from movement command /* if ( PM_SaberKataDone() ) {//we came from a bounce and cannot chain to another attack because our kata is done newmove = saberMoveData[curmove].chain_idle; } else */ if ( pm->ps->clientNum >= MAX_CLIENTS && !PM_ControlledByPlayer() && (Q_irand( 0, pm->ps->forcePowerLevel[FP_SABER_OFFENSE]-1 ) || (pm->gent&&pm->gent->enemy&&pm->gent->enemy->client&&PM_InKnockDownOnGround(&pm->gent->enemy->client->ps))//enemy knocked down, use some logic || ( pm->ps->saberAnimLevel == SS_FAST && pm->gent && pm->gent->NPC && pm->gent->NPC->rank >= RANK_LT_JG && Q_irand( 0, 1 ) ) ) )//minor change to make fast-attack users use the special attacks more {//NPCs use more randomized attacks the more skilled they are newmove = PM_NPCSaberAttackFromQuad( saberMoveData[curmove].endQuad ); } else { newmove = PM_SaberAttackForMovement( pm->cmd.forwardmove, pm->cmd.rightmove, curmove ); if ( (PM_SaberInBounce( curmove )||PM_SaberInBrokenParry( curmove )) && saberMoveData[newmove].startQuad == saberMoveData[curmove].endQuad ) {//this attack would be a repeat of the last (which was blocked), so don't actually use it, use the default chain attack for this bounce newmove = saberMoveData[curmove].chain_attack; } } if ( PM_SaberKataDone( curmove, newmove ) ) {//cannot chain this time newmove = saberMoveData[curmove].chain_idle; } } /* if ( newmove == LS_NONE ) {//FIXME: should we allow this? Are there some anims that you should never be able to chain into an attack? //only curmove that might get in here is LS_NONE, LS_DRAW, LS_PUTAWAY and the LS_R_ returns... all of which are in Q_R newmove = PM_AttackMoveForQuad( saberMoveData[curmove].endQuad ); } */ if ( newmove != LS_NONE ) { if ( !PM_InCartwheel( pm->ps->legsAnim ) ) {//don't do transitions when cartwheeling - could make you spin! //Now get the proper transition move newmove = PM_SaberAnimTransitionMove( (saberMoveName_t)curmove, (saberMoveName_t)newmove ); if ( PM_HasAnimation( pm->gent, saberMoveData[newmove].animToUse ) ) { anim = saberMoveData[newmove].animToUse; } } } } if (anim == -1) {//not side-stepping, pick neutral anim if ( !G_TryingSpecial(pm->gent,&pm->cmd)/*(pm->cmd.buttons&BUTTON_FORCE_FOCUS)*/ ) {//but only if not trying one of the special attacks! if ( PM_InCartwheel( pm->ps->legsAnim ) && pm->ps->legsAnimTimer > 100 ) {//if in the middle of a cartwheel, the chain attack is just a normal attack //NOTE: this should match the switch in PM_InCartwheel! switch( pm->ps->legsAnim ) { case BOTH_ARIAL_LEFT://swing from l to r case BOTH_CARTWHEEL_LEFT: newmove = LS_A_L2R; break; case BOTH_ARIAL_RIGHT://swing from r to l case BOTH_CARTWHEEL_RIGHT: newmove = LS_A_R2L; break; case BOTH_ARIAL_F1://random l/r attack if ( Q_irand( 0, 1 ) ) { newmove = LS_A_L2R; } else { newmove = LS_A_R2L; } break; } } else { // Add randomness for prototype? newmove = saberMoveData[curmove].chain_attack; } if ( newmove != LS_NONE ) { if (PM_HasAnimation( pm->gent, saberMoveData[newmove].animToUse)) { anim= saberMoveData[newmove].animToUse; } } } if ( !pm->cmd.forwardmove && !pm->cmd.rightmove && pm->cmd.upmove >= 0 && pm->ps->groundEntityNum != ENTITYNUM_NONE ) {//not moving at all, so set the anim on entire body both = qtrue; } } if ( anim == -1) { switch ( pm->ps->legsAnim ) { case BOTH_WALK1: case BOTH_WALK2: case BOTH_WALK_STAFF: case BOTH_WALK_DUAL: case BOTH_WALKBACK1: case BOTH_WALKBACK2: case BOTH_WALKBACK_STAFF: case BOTH_WALKBACK_DUAL: case BOTH_RUN1: case BOTH_RUN2: case BOTH_RUN_STAFF: case BOTH_RUN_DUAL: case BOTH_RUNBACK1: case BOTH_RUNBACK2: case BOTH_RUNBACK_STAFF: anim = pm->ps->legsAnim; break; default: anim = PM_ReadyPoseForSaberAnimLevel(); break; } newmove = LS_READY; } if ( !pm->ps->SaberActive() ) {//turn on the saber if it's not on pm->ps->SaberActivate(); } PM_SetSaberMove( (saberMoveName_t)newmove ); if ( both && pm->ps->legsAnim != pm->ps->torsoAnim ) { PM_SetAnim( pm,SETANIM_LEGS,pm->ps->torsoAnim,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); } if ( pm->gent && pm->gent->client ) { // pm->gent->client->saberTrail.inAction = qtrue; // pm->gent->client->saberTrail.duration = 75; // saber trail lasts for 75ms...feel free to change this if you want it longer or shorter } if ( !PM_InCartwheel( pm->ps->torsoAnim ) ) {//can still attack during a cartwheel/arial //don't fire again until anim is done pm->ps->weaponTime = pm->ps->torsoAnimTimer; } /* //FIXME: this may be making it so sometimes you can't swing again right away... if ( newmove == LS_READY ) { pm->ps->weaponTime = 500; } */ } } // ********************************************************* // WEAPON_FIRING // ********************************************************* pm->ps->weaponstate = WEAPON_FIRING; if ( pm->gent && pm->gent->client && pm->gent->client->fireDelay > 0 ) {//FIXME: this is going to fire off one frame before you expect, actually // Clear these out since we're not actually firing yet pm->ps->eFlags &= ~EF_FIRING; pm->ps->eFlags &= ~EF_ALT_FIRING; return; } addTime = pm->ps->weaponTime; /*if ( pm->cmd.buttons & BUTTON_ALT_ATTACK ) { PM_AddEvent( EV_ALT_FIRE ); if ( !addTime ) { addTime = weaponData[pm->ps->weapon].altFireTime; if ( g_timescale != NULL ) { if ( g_timescale->value < 1.0f ) { if ( !MatrixMode ) {//Special test for Matrix Mode (tm) if ( pm->ps->clientNum == 0 && !player_locked && (pm->ps->forcePowersActive&(1<ps->forcePowersActive&(1<value; } else if ( g_entities[pm->ps->clientNum].client && (pm->ps->forcePowersActive&(1<ps->forcePowersActive&(1<value; } } } } } } else */{ PM_AddEvent( EV_FIRE_WEAPON ); if ( !addTime ) { addTime = weaponData[pm->ps->weapon].fireTime; if ( g_timescale != NULL ) { if ( g_timescale->value < 1.0f ) { if ( !MatrixMode ) {//Special test for Matrix Mode (tm) if ( pm->ps->clientNum == 0 && !player_locked && (pm->ps->forcePowersActive&(1<ps->forcePowersActive&(1<value; } else if ( g_entities[pm->ps->clientNum].client && (pm->ps->forcePowersActive&(1<ps->forcePowersActive&(1<value; } } } } } } if (pm->ps->forcePowersActive & (1 << FP_RAGE)) { addTime *= 0.75; } else if (pm->ps->forceRageRecoveryTime > pm->cmd.serverTime) { addTime *= 1.5; } //If the phaser has been fired, delay the next recharge time if ( !PM_ControlledByPlayer() ) { if( pm->gent && pm->gent->NPC != NULL ) {//NPCs have their own refire logic //FIXME: this really should be universal... return; } } if ( !PM_InCartwheel( pm->ps->torsoAnim ) ) {//can still attack during a cartwheel/arial pm->ps->weaponTime = addTime; } } //--------------------------------------- static bool PM_DoChargedWeapons( void ) //--------------------------------------- { qboolean charging = qfalse, altFire = qfalse; //FIXME: make jedi aware they're being aimed at with a charged-up weapon (strafe and be evasive?) // If you want your weapon to be a charging weapon, just set this bit up switch( pm->ps->weapon ) { //------------------ case WP_BRYAR_PISTOL: case WP_BLASTER_PISTOL: // alt-fire charges the weapon if ( pm->cmd.buttons & BUTTON_ALT_ATTACK ) { charging = qtrue; altFire = qtrue; } break; //------------------ case WP_DISRUPTOR: // alt-fire charges the weapon...but due to zooming being controlled by the alt-button, the main button actually charges...but only when zoomed. // lovely, eh? if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) ) { if ( cg.zoomMode == 2 ) { if ( pm->cmd.buttons & BUTTON_ATTACK ) { charging = qtrue; altFire = qtrue; // believe it or not, it really is an alt-fire in this case! } } } else if ( pm->gent && pm->gent->NPC ) { if ( (pm->gent->NPC->scriptFlags&SCF_ALT_FIRE) ) { if ( pm->gent->fly_sound_debounce_time > level.time ) { charging = qtrue; altFire = qtrue; } } } break; //------------------ case WP_BOWCASTER: // main-fire charges the weapon if ( pm->cmd.buttons & BUTTON_ATTACK ) { charging = qtrue; } break; //------------------ case WP_DEMP2: // alt-fire charges the weapon if ( pm->cmd.buttons & BUTTON_ALT_ATTACK ) { charging = qtrue; altFire = qtrue; } break; //------------------ case WP_ROCKET_LAUNCHER: // Not really a charge weapon, but we still want to delay fire until the button comes up so that we can // implement our alt-fire locking stuff if ( pm->cmd.buttons & BUTTON_ALT_ATTACK ) { charging = qtrue; altFire = qtrue; } break; //------------------ case WP_THERMAL: // FIXME: Really should have a wind-up anim for player // as he holds down the fire button to throw, then play // the actual throw when he lets go... if ( pm->cmd.buttons & BUTTON_ALT_ATTACK ) { altFire = qtrue; // override default of not being an alt-fire charging = qtrue; } else if ( pm->cmd.buttons & BUTTON_ATTACK ) { charging = qtrue; } break; } // end switch // set up the appropriate weapon state based on the button that's down. // Note that we ALWAYS return if charging is set ( meaning the buttons are still down ) if ( charging ) { if ( altFire ) { if ( pm->ps->weaponstate != WEAPON_CHARGING_ALT && pm->ps->weaponstate != WEAPON_DROPPING ) { if ( pm->ps->ammo[weaponData[pm->ps->weapon].ammoIndex] <= 0) { PM_AddEvent( EV_NOAMMO ); pm->ps->weaponTime += 500; return true; } // charge isn't started, so do it now pm->ps->weaponstate = WEAPON_CHARGING_ALT; pm->ps->weaponChargeTime = level.time; if ( cg_weapons[pm->ps->weapon].altChargeSound ) { G_SoundOnEnt( pm->gent, CHAN_WEAPON, weaponData[pm->ps->weapon].altChargeSnd ); } } } else { if ( pm->ps->weaponstate != WEAPON_CHARGING && pm->ps->weaponstate != WEAPON_DROPPING ) { if ( pm->ps->ammo[weaponData[pm->ps->weapon].ammoIndex] <= 0) { PM_AddEvent( EV_NOAMMO ); pm->ps->weaponTime += 500; return true; } // charge isn't started, so do it now pm->ps->weaponstate = WEAPON_CHARGING; pm->ps->weaponChargeTime = level.time; if ( cg_weapons[pm->ps->weapon].chargeSound && pm->gent && !pm->gent->NPC ) // HACK: !NPC mostly for bowcaster and weequay { G_SoundOnEnt( pm->gent, CHAN_WEAPON, weaponData[pm->ps->weapon].chargeSnd ); } } } return true; // short-circuit rest of weapon code } // Only charging weapons should be able to set these states...so.... // let's see which fire mode we need to set up now that the buttons are up if ( pm->ps->weaponstate == WEAPON_CHARGING ) { // weapon has a charge, so let us do an attack // dumb, but since we shoot a charged weapon on button-up, we need to repress this button for now pm->cmd.buttons |= BUTTON_ATTACK; pm->ps->eFlags |= EF_FIRING; } else if ( pm->ps->weaponstate == WEAPON_CHARGING_ALT ) { // weapon has a charge, so let us do an alt-attack // dumb, but since we shoot a charged weapon on button-up, we need to repress this button for now pm->cmd.buttons |= BUTTON_ALT_ATTACK; pm->ps->eFlags |= (EF_FIRING|EF_ALT_FIRING); } return false; // continue with the rest of the weapon code } #define BOWCASTER_CHARGE_UNIT 200.0f // bowcaster charging gives us one more unit every 200ms--if you change this, you'll have to do the same in g_weapon #define BRYAR_CHARGE_UNIT 200.0f // bryar charging gives us one more unit every 200ms--if you change this, you'll have to do the same in g_weapon #define DEMP2_CHARGE_UNIT 500.0f // ditto #define DISRUPTOR_CHARGE_UNIT 150.0f // ditto // Specific weapons can opt to modify the ammo usage based on charges, otherwise if no special case code // is handled below, regular ammo usage will happen //--------------------------------------- static int PM_DoChargingAmmoUsage( int *amount ) //--------------------------------------- { int count = 0; if ( pm->ps->weapon == WP_BOWCASTER && !( pm->cmd.buttons & BUTTON_ALT_ATTACK )) { // this code is duplicated ( I know, I know ) in G_weapon.cpp for the bowcaster alt-fire count = ( level.time - pm->ps->weaponChargeTime ) / BOWCASTER_CHARGE_UNIT; if ( count < 1 ) { count = 1; } else if ( count > 5 ) { count = 5; } if ( !(count & 1 )) { // if we aren't odd, knock us down a level count--; } // Only bother with these checks if we don't have infinite ammo if ( pm->ps->ammo[ weaponData[pm->ps->weapon].ammoIndex ] != -1 ) { int dif = pm->ps->ammo[weaponData[pm->ps->weapon].ammoIndex] - *amount * count; // If we have enough ammo to do the full charged shot, we are ok if ( dif < 0 ) { // we are not ok, so hack our chargetime and ammo usage, note that DIF is going to be negative count += floor(dif / (float)*amount); if ( count < 1 ) { count = 1; } // now get a real chargeTime so the duplicated code in g_weapon doesn't get freaked pm->ps->weaponChargeTime = level.time - ( count * BOWCASTER_CHARGE_UNIT ); } } // now that count is cool, get the real ammo usage *amount *= count; } else if( ( pm->ps->weapon == WP_BRYAR_PISTOL && pm->cmd.buttons & BUTTON_ALT_ATTACK ) || ( pm->ps->weapon == WP_BLASTER_PISTOL && pm->cmd.buttons & BUTTON_ALT_ATTACK ) ) { // this code is duplicated ( I know, I know ) in G_weapon.cpp for the bryar alt-fire count = ( level.time - pm->ps->weaponChargeTime ) / BRYAR_CHARGE_UNIT; if ( count < 1 ) { count = 1; } else if ( count > 5 ) { count = 5; } // Only bother with these checks if we don't have infinite ammo if ( pm->ps->ammo[ weaponData[pm->ps->weapon].ammoIndex ] != -1 ) { int dif = pm->ps->ammo[weaponData[pm->ps->weapon].ammoIndex] - *amount * count; // If we have enough ammo to do the full charged shot, we are ok if ( dif < 0 ) { // we are not ok, so hack our chargetime and ammo usage, note that DIF is going to be negative count += floor(dif / (float)*amount); if ( count < 1 ) { count = 1; } // now get a real chargeTime so the duplicated code in g_weapon doesn't get freaked pm->ps->weaponChargeTime = level.time - ( count * BRYAR_CHARGE_UNIT ); } } // now that count is cool, get the real ammo usage *amount *= count; } else if ( pm->ps->weapon == WP_DEMP2 && pm->cmd.buttons & BUTTON_ALT_ATTACK ) { // this code is duplicated ( I know, I know ) in G_weapon.cpp for the demp2 alt-fire count = ( level.time - pm->ps->weaponChargeTime ) / DEMP2_CHARGE_UNIT; if ( count < 1 ) { count = 1; } else if ( count > 3 ) { count = 3; } // Only bother with these checks if we don't have infinite ammo if ( pm->ps->ammo[ weaponData[pm->ps->weapon].ammoIndex ] != -1 ) { int dif = pm->ps->ammo[weaponData[pm->ps->weapon].ammoIndex] - *amount * count; // If we have enough ammo to do the full charged shot, we are ok if ( dif < 0 ) { // we are not ok, so hack our chargetime and ammo usage, note that DIF is going to be negative count += floor(dif / (float)*amount); if ( count < 1 ) { count = 1; } // now get a real chargeTime so the duplicated code in g_weapon doesn't get freaked pm->ps->weaponChargeTime = level.time - ( count * DEMP2_CHARGE_UNIT ); } } // now that count is cool, get the real ammo usage *amount *= count; // this is an after-thought. should probably re-write the function to do this naturally. if ( *amount > pm->ps->ammo[weaponData[pm->ps->weapon].ammoIndex] ) { *amount = pm->ps->ammo[weaponData[pm->ps->weapon].ammoIndex]; } } else if ( pm->ps->weapon == WP_DISRUPTOR && pm->cmd.buttons & BUTTON_ALT_ATTACK ) // BUTTON_ATTACK will have been mapped to BUTTON_ALT_ATTACK if we are zoomed { // this code is duplicated ( I know, I know ) in G_weapon.cpp for the disruptor alt-fire count = ( level.time - pm->ps->weaponChargeTime ) / DISRUPTOR_CHARGE_UNIT; if ( count < 1 ) { count = 1; } else if ( count > 10 ) { count = 10; } // Only bother with these checks if we don't have infinite ammo if ( pm->ps->ammo[ weaponData[pm->ps->weapon].ammoIndex ] != -1 ) { int dif = pm->ps->ammo[weaponData[pm->ps->weapon].ammoIndex] - *amount * count; // If we have enough ammo to do the full charged shot, we are ok if ( dif < 0 ) { // we are not ok, so hack our chargetime and ammo usage, note that DIF is going to be negative count += floor(dif / (float)*amount); if ( count < 1 ) { count = 1; } // now get a real chargeTime so the duplicated code in g_weapon doesn't get freaked pm->ps->weaponChargeTime = level.time - ( count * DISRUPTOR_CHARGE_UNIT ); } } // now that count is cool, get the real ammo usage *amount *= count; // this is an after-thought. should probably re-write the function to do this naturally. if ( *amount > pm->ps->ammo[weaponData[pm->ps->weapon].ammoIndex] ) { *amount = pm->ps->ammo[weaponData[pm->ps->weapon].ammoIndex]; } } return count; } qboolean PM_DroidMelee( int npc_class ) { if ( npc_class == CLASS_PROBE || npc_class == CLASS_SEEKER || npc_class == CLASS_INTERROGATOR || npc_class == CLASS_SENTRY || npc_class == CLASS_REMOTE ) { return qtrue; } return qfalse; } void PM_WeaponWampa( void ) { // make weapon function if ( pm->ps->weaponTime > 0 ) { pm->ps->weaponTime -= pml.msec; if ( pm->ps->weaponTime <= 0 ) { pm->ps->weaponTime = 0; } } // check for weapon change // can't change if weapon is firing, but can change again if lowering or raising if ( pm->ps->weaponTime <= 0 || pm->ps->weaponstate != WEAPON_FIRING ) { if ( pm->ps->weapon != pm->cmd.weapon ) { PM_BeginWeaponChange( pm->cmd.weapon ); } } if ( pm->ps->weaponTime > 0 ) { return; } // change weapon if time if ( pm->ps->weaponstate == WEAPON_DROPPING ) { PM_FinishWeaponChange(); return; } if ( pm->ps->weapon == WP_SABER && (pm->cmd.buttons&BUTTON_ATTACK) && pm->ps->torsoAnim == BOTH_HANG_IDLE ) { pm->ps->SaberActivate(); pm->ps->SaberActivateTrail( 150 ); PM_SetAnim( pm, SETANIM_BOTH, BOTH_HANG_ATTACK, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); pm->ps->weaponstate = WEAPON_FIRING; pm->ps->saberBlocked = BLOCKED_NONE; pm->ps->saberMove = LS_READY; pm->ps->saberMoveNext = LS_NONE; } else if ( pm->ps->torsoAnim == BOTH_HANG_IDLE ) { pm->ps->SaberDeactivateTrail( 0 ); pm->ps->weaponstate = WEAPON_READY; pm->ps->saberMove = LS_READY; pm->ps->saberMoveNext = LS_NONE; } } /* ============== PM_Weapon Generates weapon events and modifes the weapon counter ============== */ static void PM_Weapon( void ) { int addTime, amount, trueCount = 1; qboolean delayed_fire = qfalse; if ( (pm->ps->eFlags&EF_HELD_BY_WAMPA) ) { PM_WeaponWampa(); return; } if ( pm->ps->eFlags&EF_FORCE_DRAINED ) {//being drained return; } if ( (pm->ps->forcePowersActive&(1<ps->forceDrainEntityNum < ENTITYNUM_WORLD ) {//draining return; } if (pm->ps->weapon == WP_SABER && (cg.zoomMode==3||!cg.zoomMode||pm->ps->clientNum) ) // WP_LIGHTSABER { // Separate logic for lightsaber, but not for player when zoomed PM_WeaponLightsaber(); if ( pm->gent && pm->gent->client && pm->ps->saber[0].Active() && pm->ps->saberInFlight ) {//FIXME: put saberTrail in playerState if ( pm->gent->client->ps.saberEntityState == SES_RETURNING ) {//turn off the saber trail pm->gent->client->ps.SaberDeactivateTrail( 75 ); } else {//turn on the saber trail pm->gent->client->ps.SaberActivateTrail( 150 ); } } return; } if ( PM_InKnockDown( pm->ps ) || PM_InRoll( pm->ps )) {//in knockdown if ( pm->ps->weaponTime > 0 ) { pm->ps->weaponTime -= pml.msec; if ( pm->ps->weaponTime <= 0 ) { pm->ps->weaponTime = 0; } } return; } if( pm->gent && pm->gent->client ) { if ( pm->gent->client->fireDelay > 0 ) {//FIXME: this is going to fire off one frame before you expect, actually pm->gent->client->fireDelay -= pml.msec; if(pm->gent->client->fireDelay <= 0) {//just finished delay timer if ( pm->ps->clientNum && pm->ps->weapon == WP_ROCKET_LAUNCHER ) { G_SoundOnEnt( pm->gent, CHAN_WEAPON, "sound/weapons/rocket/lock.wav" ); pm->cmd.buttons |= BUTTON_ALT_ATTACK; } pm->gent->client->fireDelay = 0; delayed_fire = qtrue; if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && pm->ps->weapon == WP_THERMAL && pm->gent->alt_fire ) { pm->cmd.buttons |= BUTTON_ALT_ATTACK; } } else { if ( pm->ps->clientNum && pm->ps->weapon == WP_ROCKET_LAUNCHER && Q_irand( 0, 1 ) ) { G_SoundOnEnt( pm->gent, CHAN_WEAPON, "sound/weapons/rocket/tick.wav" ); } } } } // don't allow attack until all buttons are up if ( pm->ps->pm_flags & PMF_RESPAWNED ) { return; } // check for dead player if ( pm->ps->stats[STAT_HEALTH] <= 0 ) { if ( pm->gent && pm->gent->client ) { pm->ps->weapon = WP_NONE; } if ( pm->gent ) { pm->gent->s.loopSound = 0; } return; } // make weapon function if ( pm->ps->weaponTime > 0 ) { pm->ps->weaponTime -= pml.msec; if ( pm->ps->weaponTime <= 0 ) { pm->ps->weaponTime = 0; } } // check for weapon change // can't change if weapon is firing, but can change again if lowering or raising if ( (pm->ps->weaponTime <= 0 || pm->ps->weaponstate != WEAPON_FIRING) && pm->ps->weaponstate != WEAPON_CHARGING_ALT && pm->ps->weaponstate != WEAPON_CHARGING) { if ( pm->ps->weapon != pm->cmd.weapon && (!pm->ps->viewEntity || pm->ps->viewEntity >= ENTITYNUM_WORLD) && !PM_DoChargedWeapons()) { PM_BeginWeaponChange( pm->cmd.weapon ); } } if ( pm->ps->weaponTime > 0 ) { return; } // change weapon if time if ( pm->ps->weaponstate == WEAPON_DROPPING ) { PM_FinishWeaponChange(); return; } if ( pm->ps->weapon == WP_NONE ) { return; } if ( pm->ps->weaponstate == WEAPON_RAISING ) { //Just selected the weapon pm->ps->weaponstate = WEAPON_IDLE; if(pm->gent && (pm->gent->s.numbergent))) { PM_SetAnim(pm,SETANIM_TORSO,BOTH_STAND1,SETANIM_FLAG_NORMAL); } else { switch(pm->ps->weapon) { case WP_BRYAR_PISTOL: case WP_BLASTER_PISTOL: if ( pm->gent && pm->gent->weaponModel[1] > 0 ) {//dual pistols //FIXME: should be a better way of detecting a dual-pistols user so it's not hardcoded to the saboteurcommando... PM_SetAnim(pm,SETANIM_TORSO,BOTH_STAND1,SETANIM_FLAG_NORMAL); } else {//single pistol PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONIDLE2,SETANIM_FLAG_NORMAL); } break; default: PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONIDLE3,SETANIM_FLAG_NORMAL); break; } } return; } if ( pm->gent ) {//ready to throw thermal again, add it if ( pm->ps->weapon == WP_THERMAL && pm->gent->weaponModel[0] == -1 ) {//add the thermal model back in our hand // remove anything if we have anything already G_RemoveWeaponModels( pm->gent ); if (weaponData[pm->ps->weapon].weaponMdl[0]) { //might be NONE, so check if it has a model G_CreateG2AttachedWeaponModel( pm->gent, weaponData[pm->ps->weapon].weaponMdl, pm->gent->handRBolt, 0 ); //make it sound like we took another one out from... uh.. somewhere... if ( cg.time > 0 ) {//this way we don't get that annoying change weapon sound every time a map starts PM_AddEvent( EV_CHANGE_WEAPON ); } } } } if ( !delayed_fire ) {//didn't just finish a fire delay if ( PM_DoChargedWeapons()) { // In some cases the charged weapon code may want us to short circuit the rest of the firing code return; } else { if ( !pm->gent->client->fireDelay//not already waiting to fire && (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer())//player && pm->ps->weapon == WP_THERMAL//holding thermal && pm->gent//gent && pm->gent->client//client && (pm->cmd.buttons&(BUTTON_ATTACK|BUTTON_ALT_ATTACK)) )//holding fire {//delay the actual firing of the missile until the anim has played some if ( PM_StandingAnim( pm->ps->legsAnim ) || pm->ps->legsAnim == BOTH_THERMAL_READY ) { PM_SetAnim( pm, SETANIM_LEGS, BOTH_THERMAL_THROW, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); } PM_SetAnim(pm,SETANIM_TORSO,BOTH_THERMAL_THROW,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_RESTART|SETANIM_FLAG_HOLD); pm->gent->client->fireDelay = 300; pm->ps->weaponstate = WEAPON_FIRING; pm->gent->alt_fire = (qboolean)(pm->cmd.buttons&BUTTON_ALT_ATTACK); return; } } } if(!delayed_fire) { if ( pm->ps->weapon == WP_MELEE && (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) ) {//melee if ( (pm->cmd.buttons&(BUTTON_ATTACK|BUTTON_ALT_ATTACK)) != (BUTTON_ATTACK|BUTTON_ALT_ATTACK) ) {//not holding both buttons if ( (pm->cmd.buttons&BUTTON_ATTACK)&&(pm->ps->pm_flags&PMF_ATTACK_HELD) ) {//held button //clear it pm->cmd.buttons &= ~BUTTON_ATTACK; } if ( (pm->cmd.buttons&BUTTON_ALT_ATTACK)&&(pm->ps->pm_flags&PMF_ALT_ATTACK_HELD) ) {//held button //clear it pm->cmd.buttons &= ~BUTTON_ALT_ATTACK; } } } // check for fire if ( !(pm->cmd.buttons & (BUTTON_ATTACK|BUTTON_ALT_ATTACK)) ) { pm->ps->weaponTime = 0; if ( pm->gent && pm->gent->client && pm->gent->client->fireDelay > 0 ) {//Still firing pm->ps->weaponstate = WEAPON_FIRING; } else if ( pm->ps->weaponstate != WEAPON_READY ) { if ( !pm->gent || !pm->gent->NPC || pm->gent->attackDebounceTime < level.time ) { pm->ps->weaponstate = WEAPON_IDLE; } } if ( pm->ps->weapon == WP_MELEE && (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && PM_KickMove( pm->ps->saberMove ) ) {//melee, not attacking, clear move pm->ps->saberMove = LS_NONE; } return; } if (pm->gent->s.m_iVehicleNum!=0) { // No Anims if on Veh } // start the animation even if out of ammo else if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_ROCKETTROOPER ) { if ( pm->gent->client->moveType == MT_FLYSWIM ) { PM_SetAnim(pm,SETANIM_TORSO,BOTH_ATTACK2,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_RESTART|SETANIM_FLAG_HOLD); } else { PM_SetAnim(pm,SETANIM_TORSO,BOTH_ATTACK1,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_RESTART|SETANIM_FLAG_HOLD); } } #ifndef BASE_SAVE_COMPAT else if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_HAZARD_TROOPER ) { // Kneel attack //-------------- if( pm->cmd.upmove == -127 ) { PM_SetAnim(pm,SETANIM_TORSO, BOTH_KNEELATTACK, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_RESTART|SETANIM_FLAG_HOLD); } else { PM_SetAnim(pm,SETANIM_TORSO,BOTH_ATTACK1,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_RESTART|SETANIM_FLAG_HOLD); } // Standing attack //----------------- } #endif else if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_ASSASSIN_DROID ) { // Crouched Attack if (PM_CrouchAnim(pm->gent->client->ps.legsAnim)) { PM_SetAnim(pm,SETANIM_TORSO,BOTH_ATTACK2,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_RESTART|SETANIM_FLAG_HOLDLESS); } // Standing Attack //----------------- else { // if (PM_StandingAnim(pm->gent->client->ps.legsAnim)) // { // PM_SetAnim(pm,SETANIM_BOTH,BOTH_ATTACK3,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_RESTART|SETANIM_FLAG_HOLDLESS); // } // else { PM_SetAnim(pm,SETANIM_TORSO,BOTH_ATTACK3,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_RESTART|SETANIM_FLAG_HOLDLESS); } } } else { switch(pm->ps->weapon) { /* case WP_SABER://1 - handed PM_SetAnim(pm,SETANIM_TORSO,BOTH_ATTACK1,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_RESTART|SETANIM_FLAG_HOLD); break; */ case WP_BRYAR_PISTOL://1-handed case WP_BLASTER_PISTOL://1-handed if ( pm->gent && pm->gent->weaponModel[1] > 0 ) {//dual pistols PM_SetAnim(pm,SETANIM_TORSO,BOTH_GUNSIT1,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_RESTART|SETANIM_FLAG_HOLD); } else {//single pistol PM_SetAnim(pm,SETANIM_TORSO,BOTH_ATTACK2,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_RESTART|SETANIM_FLAG_HOLD); } break; case WP_MELEE: // since there's no RACE_BOTS, I listed all the droids that have might have melee attacks - dmv if ( pm->gent && pm->gent->client ) { if ( PM_DroidMelee( pm->gent->client->NPC_class ) ) { if ( rand() & 1 ) PM_SetAnim(pm,SETANIM_BOTH,BOTH_MELEE1,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD); else PM_SetAnim(pm,SETANIM_BOTH,BOTH_MELEE2,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD); } else { int anim = -1; if ( (pm->ps->clientNum < MAX_CLIENTS ||PM_ControlledByPlayer()) && g_debugMelee->integer ) { if ( (pm->cmd.buttons&BUTTON_ALT_ATTACK) ) { if ( (pm->cmd.buttons&BUTTON_ATTACK) ) { PM_TryGrab(); } else if ( !(pm->ps->pm_flags&PMF_ALT_ATTACK_HELD) ) { PM_CheckKick(); } } else if ( !(pm->ps->pm_flags&PMF_ATTACK_HELD) ) { anim = PM_PickAnim( pm->gent, BOTH_MELEE1, BOTH_MELEE2 ); } } else { anim = PM_PickAnim( pm->gent, BOTH_MELEE1, BOTH_MELEE2 ); } if ( anim != -1 ) { if ( VectorCompare( pm->ps->velocity, vec3_origin ) && pm->cmd.upmove >= 0 ) { PM_SetAnim( pm, SETANIM_BOTH, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD|SETANIM_FLAG_RESTART ); } else { PM_SetAnim( pm, SETANIM_TORSO, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD|SETANIM_FLAG_RESTART ); } } } } break; case WP_TUSKEN_RIFLE: if ( pm->cmd.buttons & BUTTON_ALT_ATTACK ) {//shoot //in alt-fire, sniper mode PM_SetAnim( pm, SETANIM_TORSO, BOTH_ATTACK4, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); } else {//melee int anim = PM_PickAnim( pm->gent, BOTH_TUSKENATTACK1, BOTH_TUSKENATTACK3 ); // Rifle if ( VectorCompare( pm->ps->velocity, vec3_origin ) && pm->cmd.upmove >= 0 ) { PM_SetAnim( pm, SETANIM_BOTH, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD|SETANIM_FLAG_RESTART ); } else { PM_SetAnim( pm, SETANIM_TORSO, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD|SETANIM_FLAG_RESTART ); } } break; case WP_TUSKEN_STAFF: if ( pm->gent && pm->gent->client ) { int anim; int flags = (SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD|SETANIM_FLAG_RESTART); if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) ) {//player if ( pm->cmd.buttons & BUTTON_ALT_ATTACK ) { if ( pm->cmd.buttons & BUTTON_ATTACK ) { anim = BOTH_TUSKENATTACK3; } else { anim = BOTH_TUSKENATTACK2; } } else { anim = BOTH_TUSKENATTACK1; } } else {// npc if ( pm->cmd.buttons & BUTTON_ALT_ATTACK ) { anim = BOTH_TUSKENLUNGE1; if (pm->ps->torsoAnimTimer>0) { flags &= ~SETANIM_FLAG_RESTART; } } else { anim = PM_PickAnim( pm->gent, BOTH_TUSKENATTACK1, BOTH_TUSKENATTACK3 ); } } if ( VectorCompare( pm->ps->velocity, vec3_origin ) && pm->cmd.upmove >= 0 ) { PM_SetAnim( pm, SETANIM_BOTH, anim, flags, 0); } else { PM_SetAnim( pm, SETANIM_TORSO, anim, flags, 0); } } break; case WP_NOGHRI_STICK: if ( pm->gent && pm->gent->client ) { int anim; if ( pm->cmd.buttons & BUTTON_ATTACK ) { anim = BOTH_ATTACK3; } else { anim = PM_PickAnim( pm->gent, BOTH_TUSKENATTACK1, BOTH_TUSKENATTACK3 ); } if ( anim != BOTH_ATTACK3 && VectorCompare( pm->ps->velocity, vec3_origin ) && pm->cmd.upmove >= 0 ) { PM_SetAnim( pm, SETANIM_BOTH, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD|SETANIM_FLAG_RESTART ); } else { PM_SetAnim( pm, SETANIM_TORSO, anim, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD|SETANIM_FLAG_RESTART ); } } break; case WP_BLASTER: PM_SetAnim( pm, SETANIM_TORSO, BOTH_ATTACK3, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD|SETANIM_FLAG_RESTART); break; case WP_DISRUPTOR: if ( ((pm->ps->clientNum >= MAX_CLIENTS&&!PM_ControlledByPlayer())&& pm->gent && pm->gent->NPC && (pm->gent->NPC->scriptFlags&SCF_ALT_FIRE)) || ((pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && cg.zoomMode == 2 ) ) {//NPC or player in alt-fire, sniper mode PM_SetAnim( pm, SETANIM_TORSO, BOTH_ATTACK4, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); } else {//in primary fire mode PM_SetAnim( pm, SETANIM_TORSO, BOTH_ATTACK3, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD|SETANIM_FLAG_RESTART); } break; case WP_BOT_LASER: PM_SetAnim(pm,SETANIM_TORSO,BOTH_ATTACK1,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_RESTART|SETANIM_FLAG_HOLD); break; case WP_THERMAL: if ( (pm->ps->clientNum >= MAX_CLIENTS&&!PM_ControlledByPlayer()) ) { if ( PM_StandingAnim( pm->ps->legsAnim ) ) { PM_SetAnim( pm, SETANIM_LEGS, BOTH_ATTACK10, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); } PM_SetAnim(pm,SETANIM_TORSO,BOTH_ATTACK10,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_RESTART|SETANIM_FLAG_HOLD); } else { if ( cg.renderingThirdPerson ) { if ( PM_StandingAnim( pm->ps->legsAnim ) || pm->ps->legsAnim == BOTH_THERMAL_READY ) { PM_SetAnim( pm, SETANIM_LEGS, BOTH_THERMAL_THROW, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD ); } PM_SetAnim(pm,SETANIM_TORSO,BOTH_THERMAL_THROW,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD);//|SETANIM_FLAG_RESTART } else { PM_SetAnim(pm,SETANIM_TORSO,BOTH_ATTACK2,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_RESTART|SETANIM_FLAG_HOLD); } } break; case WP_EMPLACED_GUN: // Guess we don't play an attack animation? Maybe we should have a custom one?? break; case WP_NONE: // no anim break; case WP_REPEATER: if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_GALAKMECH ) {// if ( pm->cmd.buttons & BUTTON_ALT_ATTACK ) { PM_SetAnim(pm,SETANIM_TORSO,BOTH_ATTACK3,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_RESTART|SETANIM_FLAG_HOLD); } else { PM_SetAnim(pm,SETANIM_TORSO,BOTH_ATTACK1,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_RESTART|SETANIM_FLAG_HOLD); } } else { PM_SetAnim(pm,SETANIM_TORSO,BOTH_ATTACK3,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_RESTART|SETANIM_FLAG_HOLD); } break; case WP_TRIP_MINE: case WP_DET_PACK: PM_SetAnim(pm,SETANIM_TORSO,BOTH_ATTACK11,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_RESTART|SETANIM_FLAG_HOLD); break; default://2-handed heavy weapon PM_SetAnim(pm,SETANIM_TORSO,BOTH_ATTACK3,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_RESTART|SETANIM_FLAG_HOLD); break; } } } if ( pm->cmd.buttons & BUTTON_ALT_ATTACK ) { amount = weaponData[pm->ps->weapon].altEnergyPerShot; } else { amount = weaponData[pm->ps->weapon].energyPerShot; } if ( (pm->ps->weaponstate == WEAPON_CHARGING) || (pm->ps->weaponstate == WEAPON_CHARGING_ALT) ) { // charging weapons may want to do their own ammo logic. trueCount = PM_DoChargingAmmoUsage( &amount ); } pm->ps->weaponstate = WEAPON_FIRING; // take an ammo away if not infinite if ( pm->ps->ammo[ weaponData[pm->ps->weapon].ammoIndex ] != -1 ) { // enough energy to fire this weapon? if ((pm->ps->ammo[weaponData[pm->ps->weapon].ammoIndex] - amount) >= 0) { pm->ps->ammo[weaponData[pm->ps->weapon].ammoIndex] -= amount; } else // Not enough energy { if ( !( pm->ps->eFlags & EF_LOCKED_TO_WEAPON )) { // Switch weapons PM_AddEvent( EV_NOAMMO ); pm->ps->weaponTime += 500; } return; } } if ( pm->gent && pm->gent->client && pm->gent->client->fireDelay > 0 ) {//FIXME: this is going to fire off one frame before you expect, actually // Clear these out since we're not actually firing yet pm->ps->eFlags &= ~EF_FIRING; pm->ps->eFlags &= ~EF_ALT_FIRING; return; } if ( pm->ps->weapon == WP_EMPLACED_GUN ) { if ( pm->gent && pm->gent->owner && pm->gent->owner->e_UseFunc == useF_eweb_use ) {//eweb always shoots alt-fire, for proper effects and sounds PM_AddEvent( EV_ALT_FIRE ); addTime = weaponData[pm->ps->weapon].altFireTime; } else {//emplaced gun always shoots normal fire PM_AddEvent( EV_FIRE_WEAPON ); addTime = weaponData[pm->ps->weapon].fireTime; } } else if ( (pm->ps->weapon == WP_MELEE && (pm->ps->clientNum>=MAX_CLIENTS||!g_debugMelee->integer) ) || pm->ps->weapon == WP_TUSKEN_STAFF || (pm->ps->weapon == WP_TUSKEN_RIFLE&&!(pm->cmd.buttons&BUTTON_ALT_ATTACK)) ) { PM_AddEvent( EV_FIRE_WEAPON ); addTime = pm->ps->torsoAnimTimer; } else if ( pm->cmd.buttons & BUTTON_ALT_ATTACK ) { PM_AddEvent( EV_ALT_FIRE ); addTime = weaponData[pm->ps->weapon].altFireTime; if ( pm->ps->weapon == WP_THERMAL ) {//threw our thermal if ( pm->gent ) {// remove the thermal model if we had it. G_RemoveWeaponModels( pm->gent ); if ( (pm->ps->clientNum >= MAX_CLIENTS&&!PM_ControlledByPlayer()) ) {//NPCs need to know when to put the thermal back in their hand pm->ps->weaponTime = pm->ps->torsoAnimTimer-500; } } } } else { if ( pm->ps->clientNum //NPC && !PM_ControlledByPlayer() //not under player control && pm->ps->weapon == WP_THERMAL //using thermals && pm->ps->torsoAnim != BOTH_ATTACK10 )//not in the throw anim {//oops, got knocked out of the anim, don't throw the thermal return; } PM_AddEvent( EV_FIRE_WEAPON ); addTime = weaponData[pm->ps->weapon].fireTime; switch( pm->ps->weapon) { case WP_REPEATER: // repeater is supposed to do smoke after sustained bursts pm->ps->weaponShotCount++; break; case WP_BOWCASTER: addTime *= (( trueCount < 3 ) ? 0.35f : 1.0f );// if you only did a small charge shot with the bowcaster, use less time between shots break; case WP_THERMAL: if ( pm->gent ) {// remove the thermal model if we had it. G_RemoveWeaponModels( pm->gent ); if ( (pm->ps->clientNum >= MAX_CLIENTS&&!PM_ControlledByPlayer()) ) {//NPCs need to know when to put the thermal back in their hand pm->ps->weaponTime = pm->ps->torsoAnimTimer-500; } } break; } } if(!PM_ControlledByPlayer()) { if(pm->gent && pm->gent->NPC != NULL ) {//NPCs have their own refire logic return; } } if ( g_timescale != NULL ) { if ( g_timescale->value < 1.0f ) { if ( !MatrixMode ) {//Special test for Matrix Mode (tm) if ( pm->ps->clientNum == 0 && !player_locked && (pm->ps->forcePowersActive&(1<ps->forcePowersActive&(1<value; } else if ( g_entities[pm->ps->clientNum].client && (pm->ps->forcePowersActive&(1<ps->forcePowersActive&(1<value; } } } } pm->ps->weaponTime += addTime; pm->ps->lastShotTime = level.time;//so we know when the last time we fired our gun is // HACK!!!!! if ( pm->ps->ammo[weaponData[pm->ps->weapon].ammoIndex] <= 0 ) { if ( pm->ps->weapon == WP_THERMAL || pm->ps->weapon == WP_TRIP_MINE ) { // because these weapons have the ammo attached to the hand, we should switch weapons when the last one is thrown, otherwise it will look silly // NOTE: could also switch to an empty had version, but was told we aren't getting any new models at this point CG_OutOfAmmoChange(); PM_SetAnim(pm,SETANIM_TORSO,TORSO_DROPWEAP1 + 2,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD); // hack weapon down! pm->ps->weaponTime = 50; } } } /* ============== PM_VehicleWeapon Generates weapon events and modifes the weapon counter ============== */ static void PM_VehicleWeapon( void ) { int addTime = 0; qboolean delayed_fire = qfalse; if ( pm->ps->weapon == WP_NONE ) { return; } if(pm->gent && pm->gent->client && pm->gent->client->fireDelay > 0) {//FIXME: this is going to fire off one frame before you expect, actually pm->gent->client->fireDelay -= pml.msec; if(pm->gent->client->fireDelay <= 0) {//just finished delay timer if ( pm->ps->clientNum && pm->ps->weapon == WP_ROCKET_LAUNCHER ) { G_SoundOnEnt( pm->gent, CHAN_WEAPON, "sound/weapons/rocket/lock.wav" ); pm->cmd.buttons |= BUTTON_ALT_ATTACK; } pm->gent->client->fireDelay = 0; delayed_fire = qtrue; } else { if ( pm->ps->clientNum && pm->ps->weapon == WP_ROCKET_LAUNCHER && Q_irand( 0, 1 ) ) { G_SoundOnEnt( pm->gent, CHAN_WEAPON, "sound/weapons/rocket/tick.wav" ); } } } // don't allow attack until all buttons are up if ( pm->ps->pm_flags & PMF_RESPAWNED ) { return; } // check for dead player if ( pm->ps->stats[STAT_HEALTH] <= 0 ) { if ( pm->gent ) { pm->gent->s.loopSound = 0; } return; } // make weapon function if ( pm->ps->weaponTime > 0 ) { pm->ps->weaponTime -= pml.msec; if ( pm->ps->weaponTime <= 0 ) { pm->ps->weaponTime = 0; } } if ( pm->ps->weaponTime > 0 ) { return; } // change weapon if time if ( pm->ps->weaponstate == WEAPON_DROPPING ) { PM_FinishWeaponChange(); return; } if ( PM_DoChargedWeapons()) { // In some cases the charged weapon code may want us to short circuit the rest of the firing code return; } if(!delayed_fire) { // check for fire if ( !(pm->cmd.buttons & (BUTTON_ATTACK|BUTTON_ALT_ATTACK)) ) { pm->ps->weaponTime = 0; if ( pm->gent && pm->gent->client && pm->gent->client->fireDelay > 0 ) {//Still firing pm->ps->weaponstate = WEAPON_FIRING; } else if ( pm->ps->weaponstate != WEAPON_READY ) { if ( !pm->gent || !pm->gent->NPC || pm->gent->attackDebounceTime < level.time ) { pm->ps->weaponstate = WEAPON_IDLE; } } return; } } pm->ps->weaponstate = WEAPON_FIRING; if ( pm->gent && pm->gent->client && pm->gent->client->fireDelay > 0 ) {//FIXME: this is going to fire off one frame before you expect, actually // Clear these out since we're not actually firing yet pm->ps->eFlags &= ~EF_FIRING; pm->ps->eFlags &= ~EF_ALT_FIRING; return; } if ( pm->cmd.buttons & BUTTON_ALT_ATTACK ) { PM_AddEvent( EV_ALT_FIRE ); //addTime = weaponData[pm->ps->weapon].altFireTime; } else { PM_AddEvent( EV_FIRE_WEAPON ); // TODO: Use the real weapon fire time from the vehicle cfg file. //addTime = weaponData[pm->ps->weapon].fireTime; } /* if(pm->gent && pm->gent->NPC != NULL ) {//NPCs have their own refire logic return; }*/ if ( g_timescale != NULL ) { if ( g_timescale->value < 1.0f ) { if ( !MatrixMode ) {//Special test for Matrix Mode (tm) if ( pm->ps->clientNum == 0 && !player_locked && (pm->ps->forcePowersActive&(1<ps->forcePowersActive&(1<value; } else if ( g_entities[pm->ps->clientNum].client && (pm->ps->forcePowersActive&(1<ps->forcePowersActive&(1<value; } } } } pm->ps->weaponTime += addTime; pm->ps->lastShotTime = level.time;//so we know when the last time we fired our gun is } extern void ForceHeal( gentity_t *self ); extern void ForceTelepathy( gentity_t *self ); extern void ForceRage( gentity_t *self ); extern void ForceProtect( gentity_t *self ); extern void ForceAbsorb( gentity_t *self ); extern void ForceSeeing( gentity_t *self ); void PM_CheckForceUseButton( gentity_t *ent, usercmd_t *ucmd ) { if ( !ent ) { return; } if ( ucmd->buttons & BUTTON_USE_FORCE ) { if (!(ent->client->ps.pm_flags & PMF_USEFORCE_HELD)) { //impulse one shot switch ( showPowers[cg.forcepowerSelect] ) { case FP_HEAL: ForceHeal( ent ); break; case FP_SPEED: ForceSpeed( ent ); break; case FP_PUSH: ForceThrow( ent, qfalse ); break; case FP_PULL: ForceThrow( ent, qtrue ); break; case FP_TELEPATHY: ForceTelepathy( ent ); break; // Added 01/20/03 by AReis. // New Jedi Academy powers. case FP_RAGE: //duration - speed, invincibility and extra damage for short period, drains your health and leaves you weak and slow afterwards. ForceRage( ent ); break; case FP_PROTECT: //duration - protect against physical/energy (level 1 stops blaster/energy bolts, level 2 stops projectiles, level 3 protects against explosions) ForceProtect( ent ); break; case FP_ABSORB: //duration - protect against dark force powers (grip, lightning, drain - maybe push/pull, too?) ForceAbsorb( ent ); break; case FP_SEE: //duration - detect/see hidden enemies ForceSeeing( ent ); break; } } //these stay are okay to call every frame button is down switch ( showPowers[cg.forcepowerSelect] ) { case FP_LEVITATION: ucmd->upmove = 127; break; case FP_GRIP: ucmd->buttons |= BUTTON_FORCEGRIP; break; case FP_LIGHTNING: ucmd->buttons |= BUTTON_FORCE_LIGHTNING; break; case FP_DRAIN: // FIXME! Failing at WP_ForcePowerUsable(). -AReis ucmd->buttons |= BUTTON_FORCE_DRAIN; break; // default: // Com_Printf( "Use Force: Unhandled force: %d\n", showPowers[cg.forcepowerSelect]); // break; } ent->client->ps.pm_flags |= PMF_USEFORCE_HELD; } else//not pressing USE_FORCE { ent->client->ps.pm_flags &= ~PMF_USEFORCE_HELD; } } /* ================ PM_ForcePower ================ sends event to client for client side fx, not used */ /* static void PM_ForcePower(void) { // check for item using if ( pm->cmd.buttons & BUTTON_USE_FORCE ) { if ( ! ( pm->ps->pm_flags & PMF_USE_FORCE ) ) { pm->ps->pm_flags |= PMF_USE_FORCE; PM_AddEvent( EV_USE_FORCE); return; } } else { pm->ps->pm_flags &= ~PMF_USE_FORCE; } } */ /* ================ PM_DropTimers ================ */ static void PM_DropTimers( void ) { // drop misc timing counter if ( pm->ps->pm_time ) { if ( pml.msec >= pm->ps->pm_time ) { pm->ps->pm_flags &= ~PMF_ALL_TIMES; pm->ps->pm_time = 0; } else { pm->ps->pm_time -= pml.msec; } } // drop legs animation counter if ( pm->ps->legsAnimTimer > 0 ) { int newTime = pm->ps->legsAnimTimer - pml.msec; if ( newTime < 0 ) { newTime = 0; } PM_SetLegsAnimTimer( pm->gent, &pm->ps->legsAnimTimer, newTime ); } // drop torso animation counter if ( pm->ps->torsoAnimTimer > 0 ) { int newTime = pm->ps->torsoAnimTimer - pml.msec; if ( newTime < 0 ) { newTime = 0; } PM_SetTorsoAnimTimer( pm->gent, &pm->ps->torsoAnimTimer, newTime ); } } void PM_SetSpecialMoveValues (void ) { Flying = 0; if ( pm->gent ) { if ( pm->gent->client && pm->gent->client->moveType == MT_FLYSWIM ) { Flying = FLY_NORMAL; } else if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_VEHICLE ) { if ( pm->gent->m_pVehicle->m_pVehicleInfo->type == VH_FIGHTER ) { Flying = FLY_VEHICLE; } else if ( pm->gent->m_pVehicle->m_pVehicleInfo->hoverHeight > 0 ) {//FIXME: or just check for hoverHeight? Flying = FLY_HOVER; } } } if ( g_timescale != NULL ) { if ( g_timescale->value < 1.0f ) { if ( !MatrixMode ) { if ( pm->ps->clientNum == 0 && !player_locked && (pm->ps->forcePowersActive&(1<ps->forcePowersActive&(1<value); } else if ( g_entities[pm->ps->clientNum].client && (pm->ps->forcePowersActive&(1<ps->forcePowersActive&(1<value); } } } } } extern float cg_zoomFov; //from cg_view.cpp //------------------------------------------- void PM_AdjustAttackStates( pmove_t *pm ) //------------------------------------------- { int amount; if ( !g_saberAutoBlocking->integer && !g_saberNewControlScheme->integer && (pm->cmd.buttons&BUTTON_FORCE_FOCUS) ) { pm->ps->saberBlockingTime = pm->cmd.serverTime + 100; pm->cmd.buttons &= ~BUTTON_ATTACK; pm->cmd.buttons &= ~BUTTON_ALT_ATTACK; } // get ammo usage if ( pm->cmd.buttons & BUTTON_ALT_ATTACK ) { amount = pm->ps->ammo[weaponData[ pm->ps->weapon ].ammoIndex] - weaponData[pm->ps->weapon].altEnergyPerShot; } else { amount = pm->ps->ammo[weaponData[ pm->ps->weapon ].ammoIndex] - weaponData[pm->ps->weapon].energyPerShot; } if ( pm->ps->weapon == WP_SABER && (!cg.zoomMode||pm->ps->clientNum) ) {//don't let the alt-attack be interpreted as an actual attack command if ( pm->ps->saberInFlight ) { pm->cmd.buttons &= ~BUTTON_ALT_ATTACK; //FIXME: what about alt-attack modifier button? if ( (!pm->ps->dualSabers || !pm->ps->saber[1].Active()) ) {//saber not in hand, can't swing it pm->cmd.buttons &= ~BUTTON_ATTACK; } } //saber staff alt-attack does a special attack anim, non-throwable sabers do kicks if ( pm->ps->saberAnimLevel != SS_STAFF && !(pm->ps->saber[0].saberFlags&SFL_NOT_THROWABLE) ) {//using a throwable saber, so remove the saber throw button if ( !g_saberNewControlScheme->integer && PM_CanDoKata() ) {//old control scheme - alt-attack + attack does kata } else {//new control scheme - alt-attack doesn't have anything to do with katas, safe to clear it here pm->cmd.buttons &= ~BUTTON_ALT_ATTACK; } } } // disruptor alt-fire should toggle the zoom mode, but only bother doing this for the player? if ( pm->ps->weapon == WP_DISRUPTOR && pm->gent && (pm->gent->s.numbergent)) && pm->ps->weaponstate != WEAPON_DROPPING ) { // we are not alt-firing yet, but the alt-attack button was just pressed and // we either are ducking ( in which case we don't care if they are moving )...or they are not ducking...and also not moving right/forward. if ( !(pm->ps->eFlags & EF_ALT_FIRING) && (pm->cmd.buttons & BUTTON_ALT_ATTACK) && ( pm->cmd.upmove < 0 || ( !pm->cmd.forwardmove && !pm->cmd.rightmove ))) { // We just pressed the alt-fire key if ( cg.zoomMode == 0 || cg.zoomMode == 3 ) { G_SoundOnEnt( pm->gent, CHAN_AUTO, "sound/weapons/disruptor/zoomstart.wav" ); // not already zooming, so do it now cg.zoomMode = 2; cg.zoomLocked = qfalse; cg_zoomFov = 80.0f;//(cg.overrides.active&CG_OVERRIDE_FOV) ? cg.overrides.fov : cg_fov.value; } else if ( cg.zoomMode == 2 ) { G_SoundOnEnt( pm->gent, CHAN_AUTO, "sound/weapons/disruptor/zoomend.wav" ); // already zooming, so must be wanting to turn it off cg.zoomMode = 0; cg.zoomTime = cg.time; cg.zoomLocked = qfalse; } } else if ( !(pm->cmd.buttons & BUTTON_ALT_ATTACK )) { // Not pressing zoom any more if ( cg.zoomMode == 2 ) { // were zooming in, so now lock the zoom cg.zoomLocked = qtrue; } } if ( pm->cmd.buttons & BUTTON_ATTACK ) { // If we are zoomed, we should switch the ammo usage to the alt-fire, otherwise, we'll // just use whatever ammo was selected from above if ( cg.zoomMode == 2 ) { amount = pm->ps->ammo[weaponData[ pm->ps->weapon ].ammoIndex] - weaponData[pm->ps->weapon].altEnergyPerShot; } } else { // alt-fire button pressing doesn't use any ammo amount = 0; } } // Check for binocular specific mode if ( cg.zoomMode == 1 && pm->gent && (pm->gent->s.numbergent)) ) // { if ( pm->cmd.buttons & BUTTON_ALT_ATTACK && pm->ps->batteryCharge ) { // zooming out cg.zoomLocked = qfalse; cg.zoomDir = 1; } else if ( pm->cmd.buttons & BUTTON_ATTACK && pm->ps->batteryCharge ) { // zooming in cg.zoomLocked = qfalse; cg.zoomDir = -1; } else { // if no buttons are down, we should be in a locked state cg.zoomLocked = qtrue; } // kill buttons and associated firing flags so we can't fire pm->ps->eFlags &= ~EF_FIRING; pm->ps->eFlags &= ~EF_ALT_FIRING; pm->cmd.buttons &= ~(BUTTON_ALT_ATTACK|BUTTON_ATTACK); } // set the firing flag for continuous beam weapons, phaser will fire even if out of ammo if ( (( pm->cmd.buttons & BUTTON_ATTACK || pm->cmd.buttons & BUTTON_ALT_ATTACK ) && ( amount >= 0 || pm->ps->weapon == WP_SABER )) ) { if ( pm->cmd.buttons & BUTTON_ALT_ATTACK ) { pm->ps->eFlags |= EF_ALT_FIRING; if ( pm->ps->clientNum < MAX_CLIENTS && pm->gent && (pm->ps->eFlags&EF_IN_ATST) ) {//switch ATST barrels pm->gent->alt_fire = qtrue; } } else { pm->ps->eFlags &= ~EF_ALT_FIRING; if ( pm->ps->clientNum < MAX_CLIENTS && pm->gent && (pm->ps->eFlags&EF_IN_ATST) ) {//switch ATST barrels pm->gent->alt_fire = qfalse; } } // This flag should always get set, even when alt-firing pm->ps->eFlags |= EF_FIRING; } else { // int iFlags = pm->ps->eFlags; // Clear 'em out pm->ps->eFlags &= ~EF_FIRING; pm->ps->eFlags &= ~EF_ALT_FIRING; // if I don't check the flags before stopping FX then it switches them off too often, which tones down // the stronger FFFX so you can hardly feel them. However, if you only do iton these flags then the // repeat-fire weapons like tetrion and dreadnought don't switch off quick enough. So... // /* // Might need this for beam type weapons if ( pm->ps->weapon == WP_DREADNOUGHT || (iFlags & (EF_FIRING|EF_ALT_FIRING) ) { cgi_FF_StopAllFX(); } */ } // disruptor should convert a main fire to an alt-fire if the gun is currently zoomed if ( pm->ps->weapon == WP_DISRUPTOR && pm->gent && (pm->gent->s.numbergent)) ) { if ( pm->cmd.buttons & BUTTON_ATTACK && cg.zoomMode == 2 ) { // converting the main fire to an alt-fire pm->cmd.buttons |= BUTTON_ALT_ATTACK; pm->ps->eFlags |= EF_ALT_FIRING; } else { // don't let an alt-fire through pm->cmd.buttons &= ~BUTTON_ALT_ATTACK; } } } qboolean PM_WeaponOkOnVehicle( int weapon ) { //FIXME: check g_vehicleInfo for our vehicle? switch ( weapon ) { case WP_NONE: case WP_SABER: case WP_BLASTER: case WP_THERMAL: return qtrue; break; } return qfalse; } void PM_CheckInVehicleSaberAttackAnim( void ) {//A bit of a hack, but makes the vehicle saber attacks act like any other saber attack... // make weapon function if ( pm->ps->weaponTime > 0 ) { pm->ps->weaponTime -= pml.msec; if ( pm->ps->weaponTime <= 0 ) { pm->ps->weaponTime = 0; } } PM_CheckClearSaberBlock(); /* if ( PM_SaberBlocking() ) {//busy blocking, don't do attacks return; } */ saberMoveName_t saberMove = LS_INVALID; switch ( pm->ps->torsoAnim ) { case BOTH_VS_ATR_S: saberMove = LS_SWOOP_ATTACK_RIGHT; break; case BOTH_VS_ATL_S: saberMove = LS_SWOOP_ATTACK_LEFT; break; case BOTH_VT_ATR_S: saberMove = LS_TAUNTAUN_ATTACK_RIGHT; break; case BOTH_VT_ATL_S: saberMove = LS_TAUNTAUN_ATTACK_LEFT; break; } if ( saberMove != LS_INVALID ) { if ( pm->ps->saberMove == saberMove ) {//already playing it if ( !pm->ps->torsoAnimTimer ) {//anim was done, set it back to ready PM_SetSaberMove( LS_READY ); pm->ps->saberMove = LS_READY; pm->ps->weaponstate = WEAPON_IDLE; if (pm->cmd.buttons&BUTTON_ATTACK) { if ( !pm->ps->weaponTime ) { PM_SetSaberMove( saberMove ); pm->ps->weaponstate = WEAPON_FIRING; pm->ps->weaponTime = pm->ps->torsoAnimTimer; } } } } else if ( pm->ps->torsoAnimTimer && !pm->ps->weaponTime ) { PM_SetSaberMove( LS_READY ); pm->ps->saberMove = LS_READY; pm->ps->weaponstate = WEAPON_IDLE; PM_SetSaberMove( saberMove ); pm->ps->weaponstate = WEAPON_FIRING; pm->ps->weaponTime = pm->ps->torsoAnimTimer; } } pm->ps->saberBlocking = saberMoveData[pm->ps->saberMove].blocking; } //force the vehicle to turn and travel to its forced destination point void PM_VehForcedTurning( gentity_t *veh ) { gentity_t *dst = &g_entities[pm->ps->vehTurnaroundIndex]; float pitchD, yawD; vec3_t dir; if (!veh || !veh->m_pVehicle) { return; } if (!dst) { //can't find dest ent? return; } pm->cmd.upmove = veh->m_pVehicle->m_ucmd.upmove = 127; pm->cmd.forwardmove = veh->m_pVehicle->m_ucmd.forwardmove = 0; pm->cmd.rightmove = veh->m_pVehicle->m_ucmd.rightmove = 0; VectorSubtract(dst->s.origin, veh->currentOrigin, dir); vectoangles(dir, dir); yawD = AngleSubtract(pm->ps->viewangles[YAW], dir[YAW]); pitchD = AngleSubtract(pm->ps->viewangles[PITCH], dir[PITCH]); yawD *= 0.2f*pml.frametime; pitchD *= 0.6f*pml.frametime; pm->ps->viewangles[YAW] = AngleSubtract(pm->ps->viewangles[YAW], yawD); pm->ps->viewangles[PITCH] = AngleSubtract(pm->ps->viewangles[PITCH], pitchD); //PM_SetPMViewAngle(pm->ps, pm->ps->viewangles, &pm->cmd); SetClientViewAngle(pm->gent, pm->ps->viewangles); } /* ================ Pmove Can be called by either the server or the client ================ */ void Pmove( pmove_t *pmove ) { Vehicle_t *pVeh = NULL; pm = pmove; // this counter lets us debug movement problems with a journal by setting a conditional breakpoint fot the previous frame c_pmove++; // clear results pm->numtouch = 0; pm->watertype = 0; pm->waterlevel = 0; // Clear the blocked flag //pm->ps->pm_flags &= ~PMF_BLOCKED; pm->ps->pm_flags &= ~PMF_BUMPED; // In certain situations, we may want to control which attack buttons are pressed and what kind of functionality // is attached to them PM_AdjustAttackStates( pm ); // clear the respawned flag if attack and use are cleared if ( pm->ps->stats[STAT_HEALTH] > 0 && !( pm->cmd.buttons & BUTTON_ATTACK ) ) { pm->ps->pm_flags &= ~PMF_RESPAWNED; } // clear all pmove local vars memset (&pml, 0, sizeof(pml)); // determine the time pml.msec = pmove->cmd.serverTime - pm->ps->commandTime; if ( pml.msec < 1 ) { pml.msec = 1; } else if ( pml.msec > 200 ) { pml.msec = 200; } pm->ps->commandTime = pmove->cmd.serverTime; // save old org in case we get stuck VectorCopy (pm->ps->origin, pml.previous_origin); // save old velocity for crashlanding VectorCopy (pm->ps->velocity, pml.previous_velocity); pml.frametime = pml.msec * 0.001; if ( pm->ps->clientNum >= MAX_CLIENTS && pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_VEHICLE ) { //we are a vehicle pVeh = pm->gent->m_pVehicle; assert( pVeh ); if ( pVeh ) { pVeh->m_fTimeModifier = (pml.frametime*60.0f);//at 16.67ms (60fps), should be 1.0f } } else if ( pm->gent && PM_RidingVehicle() ) { if ( pm->ps->vehTurnaroundIndex && pm->ps->vehTurnaroundTime > pm->cmd.serverTime ) { //riding this vehicle, turn my view too PM_VehForcedTurning( &g_entities[pm->gent->s.m_iVehicleNum] ); } } PM_SetSpecialMoveValues(); // update the viewangles PM_UpdateViewAngles( pm->ps, &pm->cmd, pm->gent); AngleVectors ( pm->ps->viewangles, pml.forward, pml.right, pml.up ); if ( pm->cmd.upmove < 10 ) { // not holding jump pm->ps->pm_flags &= ~PMF_JUMP_HELD; } // decide if backpedaling animations should be used if ( pm->cmd.forwardmove < 0 ) { pm->ps->pm_flags |= PMF_BACKWARDS_RUN; } else if ( pm->cmd.forwardmove > 0 || ( pm->cmd.forwardmove == 0 && pm->cmd.rightmove ) ) { pm->ps->pm_flags &= ~PMF_BACKWARDS_RUN; } if ( pm->ps->pm_type >= PM_DEAD ) { pm->cmd.forwardmove = 0; pm->cmd.rightmove = 0; pm->cmd.upmove = 0; if ( pm->ps->viewheight > -12 ) {//slowly sink view to ground pm->ps->viewheight -= 1; } } if ( pm->ps->pm_type == PM_SPECTATOR ) { PM_CheckDuck (); PM_FlyMove (); PM_DropTimers (); return; } if ( pm->ps->pm_type == PM_NOCLIP ) { PM_NoclipMove (); PM_DropTimers (); return; } if (pm->ps->pm_type == PM_FREEZE) { return; // no movement at all } if ( pm->ps->pm_type == PM_INTERMISSION ) { return; // no movement at all } if ( pm->ps->pm_flags & PMF_SLOW_MO_FALL ) {//half grav pm->ps->gravity *= 0.5; } // set watertype, and waterlevel PM_SetWaterLevelAtPoint( pm->ps->origin, &pm->waterlevel, &pm->watertype ); PM_SetWaterHeight(); if ( !(pm->watertype & CONTENTS_LADDER) ) {//Don't want to remember this for ladders, is only for waterlevel change events (sounds) pml.previous_waterlevel = pmove->waterlevel; } waterForceJump = qfalse; if ( pmove->waterlevel && pm->ps->clientNum ) { if ( pm->ps->forceJumpZStart//force jumping ||(pm->gent&&pm->gent->NPC && level.timegent->NPC->jumpTime)) //TIMER_Done(pm->gent, "forceJumpChasing" )) )//force-jumping { waterForceJump = qtrue; } } // set mins, maxs, and viewheight PM_SetBounds(); if ( !Flying && !(pm->watertype & CONTENTS_LADDER) && pm->ps->pm_type != PM_DEAD ) {//NOTE: noclippers shouldn't jump or duck either, no? PM_CheckDuck(); } // set groundentity PM_GroundTrace(); if ( Flying == FLY_HOVER ) {//never stick to the ground PM_HoverTrace(); } if ( pm->ps->pm_type == PM_DEAD ) { PM_DeadMove (); } PM_DropTimers(); /* if ( PM_RidingVehicle() ) { PM_NoclipMove(); } else */if ( pm->ps && ( (pm->ps->eFlags&EF_LOCKED_TO_WEAPON) || (pm->ps->eFlags&EF_HELD_BY_RANCOR) || (pm->ps->eFlags&EF_HELD_BY_WAMPA) || (pm->ps->eFlags&EF_HELD_BY_SAND_CREATURE) ) ) {//in an emplaced gun PM_NoclipMove(); } else if ( Flying == FLY_NORMAL )//|| pm->ps->gravity <= 0 ) { // flight powerup doesn't allow jump and has different friction PM_FlyMove(); } else if ( Flying == FLY_VEHICLE ) { PM_FlyVehicleMove(); } else if ( pm->ps->pm_flags & PMF_TIME_WATERJUMP ) { PM_WaterJumpMove(); } else if ( pm->waterlevel > 1 //in water &&((pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) || !waterForceJump) )//player or NPC not force jumping {//force-jumping NPCs should // swimming or in ladder PM_WaterMove(); } else if (pm->gent && pm->gent->NPC && pm->gent->NPC->jumpTime!=0) { ucmd.forwardmove = 0; ucmd.rightmove = 0; ucmd.upmove = 0; pm->ps->speed = 0; VectorClear(pm->ps->moveDir); PM_AirMove(); } else if ( pml.walking ) {// walking on ground vec3_t oldOrg; VectorCopy( pm->ps->origin, oldOrg ); PM_WalkMove(); float threshHold = 0.001f, movedDist = DistanceSquared( oldOrg, pm->ps->origin ); if ( PM_StandingAnim( pm->ps->legsAnim ) || pm->ps->legsAnim == BOTH_CROUCH1 ) { threshHold = 0.005f; } if ( movedDist < threshHold ) {//didn't move, play no legs anim // pm->cmd.forwardmove = pm->cmd.rightmove = 0; } } else { if ( pm->ps->gravity <= 0 ) { PM_FlyMove(); } else { // airborne PM_AirMove(); } } //PM_Animate(); // If we didn't move at all, then why bother doing this again -MW. if(!(VectorCompare(pm->ps->origin,pml.previous_origin))) { PM_GroundTrace(); if ( Flying == FLY_HOVER ) {//never stick to the ground PM_HoverTrace(); } } if ( pm->ps->groundEntityNum != ENTITYNUM_NONE ) {//on ground pm->ps->forceJumpZStart = 0; pm->ps->jumpZStart = 0; pm->ps->pm_flags &= ~PMF_JUMPING; pm->ps->pm_flags &= ~PMF_TRIGGER_PUSHED; pm->ps->pm_flags &= ~PMF_SLOW_MO_FALL; } // If we didn't move at all, then why bother doing this again -MW. // Note: ok, so long as we don't have water levels that change. if(!(VectorCompare(pm->ps->origin,pml.previous_origin))) { PM_SetWaterLevelAtPoint( pm->ps->origin, &pm->waterlevel, &pm->watertype ); PM_SetWaterHeight(); } // PM_ForcePower(); sends event to client for client side fx, not used // If we're a vehicle, do our special weapon function. if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_VEHICLE ) { pVeh = pm->gent->m_pVehicle; // Using vehicle weapon... //if ( pm->cmd.weapon == WP_NONE ) { //PM_Weapon(); //PM_AddEvent( EV_FIRE_WEAPON ); PM_VehicleWeapon(); } } // If we are riding a vehicle... else if ( PM_RidingVehicle() ) { if ( pm->cmd.buttons & BUTTON_ALT_ATTACK ) {// alt attack always does other stuff when riding a vehicle (turbo) } else if ( (pm->ps->eFlags&EF_NODRAW) ) {//inside a vehicle? don't do any weapon stuff } else if ( pm->ps->weapon == WP_BLASTER//using blaster || pm->ps->weapon == WP_THERMAL//using thermal || pm->ps->weaponstate == WEAPON_DROPPING//changing weapon - dropping || pm->ps->weaponstate == WEAPON_RAISING//changing weapon - raising || (pm->cmd.weapon != pm->ps->weapon && PM_WeaponOkOnVehicle( pm->cmd.weapon )) )//FIXME: make this a vehicle call to see if this new weapon is valid for this vehicle {//either weilding a weapon we can fire with normal weapon logic, or trying to change to a valid weapon // call normal weapons code... should we override the normal fire anims with vehicle fire anims in here or in a subsequent call to VehicleWeapons or something? //Maybe break PM_Weapon into PM_Weapon and PM_WeaponAnimate (then call our own PM_VehicleWeaponAnimate)? PM_Weapon(); } //BUT: now call Vehicle's weapon code, to handle lightsaber and (maybe) overriding weapon ready/firing anims? } // otherwise do the normal weapon function. else { // weapons PM_Weapon(); } if ( pm->cmd.buttons & BUTTON_ATTACK ) { pm->ps->pm_flags |= PMF_ATTACK_HELD; } else { pm->ps->pm_flags &= ~PMF_ATTACK_HELD; } if ( pm->cmd.buttons & BUTTON_ALT_ATTACK ) { pm->ps->pm_flags |= PMF_ALT_ATTACK_HELD; } else { pm->ps->pm_flags &= ~PMF_ALT_ATTACK_HELD; } if ( pm->cmd.buttons & BUTTON_FORCE_FOCUS ) { pm->ps->pm_flags |= PMF_FORCE_FOCUS_HELD; } else { pm->ps->pm_flags &= ~PMF_FORCE_FOCUS_HELD; } if ( pm->gent )//&& pm->gent->s.number == 0 )//player only? { // Use PM_Use(); } // Calculate the resulting speed of the last pmove //------------------------------------------------- if ( pm->gent ) { pm->gent->resultspeed = ((Distance(pm->ps->origin, pm->gent->currentOrigin) / pml.msec) * 1000); if (pm->gent->resultspeed>5.0f) { pm->gent->lastMoveTime = level.time; } // If Have Not Been Moving For A While, Stop //------------------------------------------- if (pml.walking && (level.time - pm->gent->lastMoveTime)>1000) { pm->cmd.forwardmove = pm->cmd.rightmove = 0; } } // ANIMATION //================================ // TEMPTEMPTEMPTEMPTEMPTEMPTEMPTEMPTEMPTEMPTEMPTEMPTEMPTEMPTEMP if ( pm->gent && pm->ps && pm->ps->eFlags & EF_LOCKED_TO_WEAPON ) { if ( pm->gent->owner && pm->gent->owner->e_UseFunc == useF_emplaced_gun_use )//ugly way to tell, but... {//full body PM_SetAnim(pm,SETANIM_BOTH,BOTH_GUNSIT1,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD);//SETANIM_FLAG_NORMAL } else {//stand (or could be overridden by strafe anims) PM_SetAnim(pm,SETANIM_LEGS,BOTH_STAND1,SETANIM_FLAG_NORMAL); } } else if ( pm->gent && pm->ps && (pm->ps->eFlags&EF_HELD_BY_RANCOR) ) { PM_SetAnim(pm,SETANIM_LEGS,BOTH_SWIM_IDLE1,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD);//SETANIM_FLAG_NORMAL } // If we are a vehicle, animate... else if ( pVeh ) { pVeh->m_pVehicleInfo->Animate( pVeh ); } // If we're riding a vehicle, don't do anything!. else if ( ( pVeh = PM_RidingVehicle() ) != 0 ) { PM_CheckInVehicleSaberAttackAnim(); } else // TEMPTEMPTEMPTEMPTEMPTEMPTEMPTEMPTEMPTEMPTEMPTEMPTEMP { // footstep events / legs animations PM_Footsteps(); } // torso animation if ( !pVeh ) {//not riding a vehicle PM_TorsoAnimation(); } // entering / leaving water splashes PM_WaterEvents(); // snap some parts of playerstate to save network bandwidth // SnapVector( pm->ps->velocity ); if ( !pm->cmd.rightmove && !pm->cmd.forwardmove && pm->cmd.upmove <= 0 ) { if ( VectorCompare( pm->ps->velocity, vec3_origin ) ) { pm->ps->lastStationary = level.time; } } if ( pm->ps->pm_flags & PMF_SLOW_MO_FALL ) {//half grav pm->ps->gravity *= 2; } }