mirror of
https://github.com/ioquake/jedi-academy.git
synced 2024-11-22 12:22:23 +00:00
7206 lines
210 KiB
C++
7206 lines
210 KiB
C++
// this include must remain at the top of every bg_xxxx CPP file
|
|
#include "common_headers.h"
|
|
|
|
|
|
// 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 "../renderer/tr_local.h"
|
|
|
|
#include "q_shared.h"
|
|
#include "g_shared.h"
|
|
#include "bg_local.h"
|
|
#include "../cgame/cg_local.h"
|
|
#include "anims.h"
|
|
#include "Q3_Interface.h"
|
|
#include "g_local.h"
|
|
#include "wp_saber.h"
|
|
#include "g_vehicles.h"
|
|
|
|
extern pmove_t *pm;
|
|
extern pml_t pml;
|
|
extern cvar_t *g_ICARUSDebug;
|
|
extern cvar_t *g_timescale;
|
|
extern cvar_t *g_synchSplitAnims;
|
|
extern cvar_t *g_AnimWarning;
|
|
extern cvar_t *g_noFootSlide;
|
|
extern cvar_t *g_noFootSlideRunScale;
|
|
extern cvar_t *g_noFootSlideWalkScale;
|
|
extern cvar_t *g_saberAnimSpeed;
|
|
extern cvar_t *g_saberAutoAim;
|
|
extern cvar_t *g_speederControlScheme;
|
|
extern cvar_t *g_saberNewControlScheme;
|
|
|
|
extern qboolean InFront( vec3_t spot, vec3_t from, vec3_t fromAngles, float threshHold = 0.0f );
|
|
extern void WP_ForcePowerDrain( gentity_t *self, forcePowers_t forcePower, int overrideAmt );
|
|
extern qboolean ValidAnimFileIndex ( int index );
|
|
extern qboolean PM_ControlledByPlayer( void );
|
|
extern qboolean PM_DroidMelee( int npc_class );
|
|
extern qboolean PM_PainAnim( int anim );
|
|
extern qboolean PM_JumpingAnim( int anim );
|
|
extern qboolean PM_FlippingAnim( int anim );
|
|
extern qboolean PM_RollingAnim( int anim );
|
|
extern qboolean PM_SwimmingAnim( int anim );
|
|
extern qboolean PM_InKnockDown( playerState_t *ps );
|
|
extern qboolean PM_InRoll( playerState_t *ps );
|
|
extern qboolean PM_DodgeAnim( int anim );
|
|
extern qboolean PM_InSlopeAnim( int anim );
|
|
extern qboolean PM_ForceAnim( int anim );
|
|
extern qboolean PM_InKnockDownOnGround( playerState_t *ps );
|
|
extern qboolean PM_InSpecialJump( int anim );
|
|
extern qboolean PM_RunningAnim( int anim );
|
|
extern qboolean PM_WalkingAnim( int anim );
|
|
extern qboolean PM_SwimmingAnim( int anim );
|
|
extern qboolean PM_JumpingAnim( int anim );
|
|
extern qboolean PM_SaberStanceAnim( int anim );
|
|
extern qboolean PM_SaberDrawPutawayAnim( int anim );
|
|
extern void PM_SetJumped( float height, qboolean force );
|
|
extern qboolean PM_InGetUpNoRoll( playerState_t *ps );
|
|
extern qboolean PM_CrouchAnim( 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 qboolean G_TryingLungeAttack( gentity_t *self, usercmd_t *cmd );
|
|
extern qboolean G_TryingPullAttack( gentity_t *self, usercmd_t *cmd, qboolean amPulling );
|
|
extern qboolean G_InCinematicSaberAnim( gentity_t *self );
|
|
extern qboolean G_ControlledByPlayer( gentity_t *self );
|
|
|
|
extern int g_crosshairEntNum;
|
|
|
|
int PM_AnimLength( int index, animNumber_t anim );
|
|
qboolean PM_LockedAnim( int anim );
|
|
qboolean PM_StandingAnim( int anim );
|
|
qboolean PM_InOnGroundAnim ( playerState_t *ps );
|
|
qboolean PM_SuperBreakWinAnim( int anim );
|
|
qboolean PM_SuperBreakLoseAnim( int anim );
|
|
qboolean PM_LockedAnim( int anim );
|
|
saberMoveName_t PM_SaberFlipOverAttackMove( void );
|
|
qboolean PM_CheckFlipOverAttackMove( qboolean checkEnemy );
|
|
saberMoveName_t PM_SaberJumpForwardAttackMove( void );
|
|
qboolean PM_CheckJumpForwardAttackMove( void );
|
|
saberMoveName_t PM_SaberBackflipAttackMove( void );
|
|
qboolean PM_CheckBackflipAttackMove( void );
|
|
saberMoveName_t PM_SaberDualJumpAttackMove( void );
|
|
qboolean PM_CheckDualJumpAttackMove( void );
|
|
saberMoveName_t PM_SaberLungeAttackMove( qboolean fallbackToNormalLunge );
|
|
qboolean PM_CheckLungeAttackMove( void );
|
|
// Okay, here lies the much-dreaded Pat-created FSM movement chart... Heretic II strikes again!
|
|
// Why am I inflicting this on you? Well, it's better than hardcoded states.
|
|
// Ideally this will be replaced with an external file or more sophisticated move-picker
|
|
// once the game gets out of prototype stage.
|
|
|
|
// Silly, but I'm replacing these macros so they are shorter!
|
|
#define AFLAG_IDLE (SETANIM_FLAG_NORMAL)
|
|
#define AFLAG_ACTIVE (SETANIM_FLAG_OVERRIDE | SETANIM_FLAG_HOLD | SETANIM_FLAG_HOLDLESS)
|
|
#define AFLAG_WAIT (SETANIM_FLAG_HOLD | SETANIM_FLAG_HOLDLESS)
|
|
#define AFLAG_FINISH (SETANIM_FLAG_HOLD)
|
|
|
|
//FIXME: add the alternate anims for each style?
|
|
saberMoveData_t saberMoveData[LS_MOVE_MAX] = {// NB:randomized
|
|
// name anim(do all styles?)startQ endQ setanimflag blend, blocking chain_idle chain_attack trailLen
|
|
{"None", BOTH_STAND1, Q_R, Q_R, AFLAG_IDLE, 350, BLK_NO, LS_NONE, LS_NONE, 0 }, // LS_NONE = 0,
|
|
|
|
// General movements with saber
|
|
{"Ready", BOTH_STAND2, Q_R, Q_R, AFLAG_IDLE, 350, BLK_WIDE, LS_READY, LS_S_R2L, 0 }, // LS_READY,
|
|
{"Draw", BOTH_STAND1TO2, Q_R, Q_R, AFLAG_FINISH, 350, BLK_NO, LS_READY, LS_S_R2L, 0 }, // LS_DRAW,
|
|
{"Putaway", BOTH_STAND2TO1, Q_R, Q_R, AFLAG_FINISH, 350, BLK_NO, LS_READY, LS_S_R2L, 0 }, // LS_PUTAWAY,
|
|
|
|
// Attacks
|
|
//UL2LR
|
|
{"TL2BR Att", BOTH_A1_TL_BR, Q_TL, Q_BR, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_R_TL2BR, LS_R_TL2BR, 200 }, // LS_A_TL2BR
|
|
//SLASH LEFT
|
|
{"L2R Att", BOTH_A1__L__R, Q_L, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_R_L2R, LS_R_L2R, 200 }, // LS_A_L2R
|
|
//LL2UR
|
|
{"BL2TR Att", BOTH_A1_BL_TR, Q_BL, Q_TR, AFLAG_ACTIVE, 50, BLK_TIGHT, LS_R_BL2TR, LS_R_BL2TR, 200 }, // LS_A_BL2TR
|
|
//LR2UL
|
|
{"BR2TL Att", BOTH_A1_BR_TL, Q_BR, Q_TL, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_R_BR2TL, LS_R_BR2TL, 200 }, // LS_A_BR2TL
|
|
//SLASH RIGHT
|
|
{"R2L Att", BOTH_A1__R__L, Q_R, Q_L, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_R_R2L, LS_R_R2L, 200 },// LS_A_R2L
|
|
//UR2LL
|
|
{"TR2BL Att", BOTH_A1_TR_BL, Q_TR, Q_BL, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_R_TR2BL, LS_R_TR2BL, 200 }, // LS_A_TR2BL
|
|
//SLASH DOWN
|
|
{"T2B Att", BOTH_A1_T__B_, Q_T, Q_B, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_R_T2B, LS_R_T2B, 200 }, // LS_A_T2B
|
|
//special attacks
|
|
{"Back Stab", BOTH_A2_STABBACK1, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_READY, 200 }, // LS_A_BACKSTAB
|
|
{"Back Att", BOTH_ATTACK_BACK, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_READY, 200 }, // LS_A_BACK
|
|
{"CR Back Att", BOTH_CROUCHATTACKBACK1,Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_READY, 200 }, // LS_A_BACK_CR
|
|
{"RollStab", BOTH_ROLL_STAB, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_READY, 200 }, // LS_ROLL_STAB
|
|
{"Lunge Att", BOTH_LUNGE2_B__T_, Q_B, Q_T, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_READY, 200 }, // LS_A_LUNGE
|
|
{"Jump Att", BOTH_FORCELEAP2_T__B_,Q_T, Q_B, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_READY, 200 }, // LS_A_JUMP_T__B_
|
|
{"Flip Stab", BOTH_JUMPFLIPSTABDOWN,Q_R, Q_T, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_T1_T___R, 200 }, // LS_A_FLIP_STAB
|
|
{"Flip Slash", BOTH_JUMPFLIPSLASHDOWN1,Q_L,Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_T1__R_T_, 200 }, // LS_A_FLIP_SLASH
|
|
{"DualJump Atk",BOTH_JUMPATTACK6, Q_R, Q_BL, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_T1_BL_TR, 200 }, // LS_JUMPATTACK_DUAL
|
|
|
|
{"DualJumpAtkL_A",BOTH_ARIAL_LEFT, Q_R, Q_TL, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_A_TL2BR, 200 }, // LS_JUMPATTACK_ARIAL_LEFT
|
|
{"DualJumpAtkR_A",BOTH_ARIAL_RIGHT, Q_R, Q_TR, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_A_TR2BL, 200 }, // LS_JUMPATTACK_ARIAL_RIGHT
|
|
|
|
{"DualJumpAtkL_A",BOTH_CARTWHEEL_LEFT, Q_R,Q_TL, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_T1_TL_BR, 200 }, // LS_JUMPATTACK_CART_LEFT
|
|
{"DualJumpAtkR_A",BOTH_CARTWHEEL_RIGHT, Q_R,Q_TR, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_T1_TR_BL, 200 }, // LS_JUMPATTACK_CART_RIGHT
|
|
|
|
{"DualJumpAtkLStaff", BOTH_BUTTERFLY_FL1,Q_R,Q_L, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_T1__L__R, 200 }, // LS_JUMPATTACK_STAFF_LEFT
|
|
{"DualJumpAtkRStaff", BOTH_BUTTERFLY_FR1,Q_R,Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_T1__R__L, 200 }, // LS_JUMPATTACK_STAFF_RIGHT
|
|
|
|
{"ButterflyLeft", BOTH_BUTTERFLY_LEFT,Q_R,Q_L, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_T1__L__R, 200 }, // LS_BUTTERFLY_LEFT
|
|
{"ButterflyRight", BOTH_BUTTERFLY_RIGHT,Q_R,Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_T1__R__L, 200 }, // LS_BUTTERFLY_RIGHT
|
|
|
|
{"BkFlip Atk", BOTH_JUMPATTACK7, Q_B, Q_T, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_T1_T___R, 200 }, // LS_A_BACKFLIP_ATK
|
|
{"DualSpinAtk", BOTH_SPINATTACK6, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_READY, 200 }, // LS_SPINATTACK_DUAL
|
|
{"StfSpinAtk", BOTH_SPINATTACK7, Q_L, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_READY, 200 }, // LS_SPINATTACK
|
|
{"LngLeapAtk", BOTH_FORCELONGLEAP_ATTACK,Q_R,Q_L, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_READY, 200 }, // LS_LEAP_ATTACK
|
|
{"SwoopAtkR", BOTH_VS_ATR_S, Q_R, Q_T, AFLAG_ACTIVE, 100, BLK_NO, LS_READY, LS_READY, 200 }, // LS_SWOOP_ATTACK_RIGHT
|
|
{"SwoopAtkL", BOTH_VS_ATL_S, Q_L, Q_T, AFLAG_ACTIVE, 100, BLK_NO, LS_READY, LS_READY, 200 }, // LS_SWOOP_ATTACK_LEFT
|
|
{"TauntaunAtkR",BOTH_VT_ATR_S, Q_R, Q_T, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_READY, 200 }, // LS_TAUNTAUN_ATTACK_RIGHT
|
|
{"TauntaunAtkL",BOTH_VT_ATL_S, Q_L, Q_T, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_READY, 200 }, // LS_TAUNTAUN_ATTACK_LEFT
|
|
{"StfKickFwd", BOTH_A7_KICK_F, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_S_R2L, 200 }, // LS_KICK_F
|
|
{"StfKickBack", BOTH_A7_KICK_B, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_S_R2L, 200 }, // LS_KICK_B
|
|
{"StfKickRight",BOTH_A7_KICK_R, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_S_R2L, 200 }, // LS_KICK_R
|
|
{"StfKickLeft", BOTH_A7_KICK_L, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_S_R2L, 200 }, // LS_KICK_L
|
|
{"StfKickSpin", BOTH_A7_KICK_S, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_NO, LS_READY, LS_S_R2L, 200 }, // LS_KICK_S
|
|
{"StfKickBkFwd",BOTH_A7_KICK_BF, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_NO, LS_READY, LS_S_R2L, 200 }, // LS_KICK_BF
|
|
{"StfKickSplit",BOTH_A7_KICK_RL, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_NO, LS_READY, LS_S_R2L, 200 }, // LS_KICK_RL
|
|
{"StfKickFwdAir",BOTH_A7_KICK_F_AIR,Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_S_R2L, 200 }, // LS_KICK_F_AIR
|
|
{"StfKickBackAir",BOTH_A7_KICK_B_AIR,Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_S_R2L, 200 }, // LS_KICK_B_AIR
|
|
{"StfKickRightAir",BOTH_A7_KICK_R_AIR,Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_S_R2L, 200 }, // LS_KICK_R_AIR
|
|
{"StfKickLeftAir",BOTH_A7_KICK_L_AIR,Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_S_R2L, 200 }, // LS_KICK_L_AIR
|
|
{"StabDown", BOTH_STABDOWN, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_S_R2L, 200 }, // LS_STABDOWN
|
|
{"StabDownStf", BOTH_STABDOWN_STAFF,Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_S_R2L, 200 }, // LS_STABDOWN_STAFF
|
|
{"StabDownDual",BOTH_STABDOWN_DUAL, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_S_R2L, 200 }, // LS_STABDOWN_DUAL
|
|
{"dualspinprot",BOTH_A6_SABERPROTECT,Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_READY, 500 }, // LS_DUAL_SPIN_PROTECT
|
|
{"StfSoulCal", BOTH_A7_SOULCAL, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_READY, 500 }, // LS_STAFF_SOULCAL
|
|
{"specialfast", BOTH_A1_SPECIAL, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_READY, 2000}, // LS_A1_SPECIAL
|
|
{"specialmed", BOTH_A2_SPECIAL, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_READY, 2000}, // LS_A2_SPECIAL
|
|
{"specialstr", BOTH_A3_SPECIAL, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_READY, 2000}, // LS_A3_SPECIAL
|
|
{"upsidedwnatk",BOTH_FLIP_ATTACK7, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_READY, 200}, // LS_UPSIDE_DOWN_ATTACK
|
|
{"pullatkstab", BOTH_PULL_IMPALE_STAB,Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_READY, 200}, // LS_PULL_ATTACK_STAB
|
|
{"pullatkswing",BOTH_PULL_IMPALE_SWING,Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_READY, 200}, // LS_PULL_ATTACK_SWING
|
|
{"AloraSpinAtk",BOTH_ALORA_SPIN_SLASH,Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_READY, 200 }, // LS_SPINATTACK_ALORA
|
|
{"Dual FB Atk", BOTH_A6_FB, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_READY, 200 }, // LS_DUAL_FB
|
|
{"Dual LR Atk", BOTH_A6_LR, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_READY, 200 }, // LS_DUAL_LR
|
|
{"StfHiltBash", BOTH_A7_HILT, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_READY, LS_READY, 200 }, // LS_HILT_BASH
|
|
|
|
//starts
|
|
{"TL2BR St", BOTH_S1_S1_TL, Q_R, Q_TL, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_A_TL2BR, LS_A_TL2BR, 200 }, // LS_S_TL2BR
|
|
{"L2R St", BOTH_S1_S1__L, Q_R, Q_L, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_A_L2R, LS_A_L2R, 200 }, // LS_S_L2R
|
|
{"BL2TR St", BOTH_S1_S1_BL, Q_R, Q_BL, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_A_BL2TR, LS_A_BL2TR, 200 }, // LS_S_BL2TR
|
|
{"BR2TL St", BOTH_S1_S1_BR, Q_R, Q_BR, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_A_BR2TL, LS_A_BR2TL, 200 }, // LS_S_BR2TL
|
|
{"R2L St", BOTH_S1_S1__R, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_A_R2L, LS_A_R2L, 200 }, // LS_S_R2L
|
|
{"TR2BL St", BOTH_S1_S1_TR, Q_R, Q_TR, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_A_TR2BL, LS_A_TR2BL, 200 }, // LS_S_TR2BL
|
|
{"T2B St", BOTH_S1_S1_T_, Q_R, Q_T, AFLAG_ACTIVE, 100, BLK_TIGHT, LS_A_T2B, LS_A_T2B, 200 }, // LS_S_T2B
|
|
|
|
//returns
|
|
{"TL2BR Ret", BOTH_R1_BR_S1, Q_BR, Q_R, AFLAG_FINISH, 100, BLK_TIGHT, LS_READY, LS_READY, 200 }, // LS_R_TL2BR
|
|
{"L2R Ret", BOTH_R1__R_S1, Q_R, Q_R, AFLAG_FINISH, 100, BLK_TIGHT, LS_READY, LS_READY, 200 }, // LS_R_L2R
|
|
{"BL2TR Ret", BOTH_R1_TR_S1, Q_TR, Q_R, AFLAG_FINISH, 100, BLK_TIGHT, LS_READY, LS_READY, 200 }, // LS_R_BL2TR
|
|
{"BR2TL Ret", BOTH_R1_TL_S1, Q_TL, Q_R, AFLAG_FINISH, 100, BLK_TIGHT, LS_READY, LS_READY, 200 }, // LS_R_BR2TL
|
|
{"R2L Ret", BOTH_R1__L_S1, Q_L, Q_R, AFLAG_FINISH, 100, BLK_TIGHT, LS_READY, LS_READY, 200 }, // LS_R_R2L
|
|
{"TR2BL Ret", BOTH_R1_BL_S1, Q_BL, Q_R, AFLAG_FINISH, 100, BLK_TIGHT, LS_READY, LS_READY, 200 }, // LS_R_TR2BL
|
|
{"T2B Ret", BOTH_R1_B__S1, Q_B, Q_R, AFLAG_FINISH, 100, BLK_TIGHT, LS_READY, LS_READY, 200 }, // LS_R_T2B
|
|
|
|
//Transitions
|
|
{"BR2R Trans", BOTH_T1_BR__R, Q_BR, Q_R, AFLAG_ACTIVE, 100, BLK_NO, LS_R_L2R, LS_A_R2L, 150 }, //# Fast arc bottom right to right
|
|
{"BR2TR Trans", BOTH_T1_BR_TR, Q_BR, Q_TR, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BL2TR, LS_A_TR2BL, 150 }, //# Fast arc bottom right to top right (use: BOTH_T1_TR_BR)
|
|
{"BR2T Trans", BOTH_T1_BR_T_, Q_BR, Q_T, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BL2TR, LS_A_T2B, 150 }, //# Fast arc bottom right to top (use: BOTH_T1_T__BR)
|
|
{"BR2TL Trans", BOTH_T1_BR_TL, Q_BR, Q_TL, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BR2TL, LS_A_TL2BR, 150 }, //# Fast weak spin bottom right to top left
|
|
{"BR2L Trans", BOTH_T1_BR__L, Q_BR, Q_L, AFLAG_ACTIVE, 100, BLK_NO, LS_R_R2L, LS_A_L2R, 150 }, //# Fast weak spin bottom right to left
|
|
{"BR2BL Trans", BOTH_T1_BR_BL, Q_BR, Q_BL, AFLAG_ACTIVE, 100, BLK_NO, LS_R_TR2BL, LS_A_BL2TR, 150 }, //# Fast weak spin bottom right to bottom left
|
|
{"R2BR Trans", BOTH_T1__R_BR, Q_R, Q_BR, AFLAG_ACTIVE, 100, BLK_NO, LS_R_TL2BR, LS_A_BR2TL, 150 }, //# Fast arc right to bottom right (use: BOTH_T1_BR__R)
|
|
{"R2TR Trans", BOTH_T1__R_TR, Q_R, Q_TR, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BL2TR, LS_A_TR2BL, 150 }, //# Fast arc right to top right
|
|
{"R2T Trans", BOTH_T1__R_T_, Q_R, Q_T, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BL2TR, LS_A_T2B, 150 }, //# Fast ar right to top (use: BOTH_T1_T___R)
|
|
{"R2TL Trans", BOTH_T1__R_TL, Q_R, Q_TL, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BR2TL, LS_A_TL2BR, 150 }, //# Fast arc right to top left
|
|
{"R2L Trans", BOTH_T1__R__L, Q_R, Q_L, AFLAG_ACTIVE, 100, BLK_NO, LS_R_R2L, LS_A_L2R, 150 }, //# Fast weak spin right to left
|
|
{"R2BL Trans", BOTH_T1__R_BL, Q_R, Q_BL, AFLAG_ACTIVE, 100, BLK_NO, LS_R_TR2BL, LS_A_BL2TR, 150 }, //# Fast weak spin right to bottom left
|
|
{"TR2BR Trans", BOTH_T1_TR_BR, Q_TR, Q_BR, AFLAG_ACTIVE, 100, BLK_NO, LS_R_TL2BR, LS_A_BR2TL, 150 }, //# Fast arc top right to bottom right
|
|
{"TR2R Trans", BOTH_T1_TR__R, Q_TR, Q_R, AFLAG_ACTIVE, 100, BLK_NO, LS_R_L2R, LS_A_R2L, 150 }, //# Fast arc top right to right (use: BOTH_T1__R_TR)
|
|
{"TR2T Trans", BOTH_T1_TR_T_, Q_TR, Q_T, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BL2TR, LS_A_T2B, 150 }, //# Fast arc top right to top (use: BOTH_T1_T__TR)
|
|
{"TR2TL Trans", BOTH_T1_TR_TL, Q_TR, Q_TL, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BR2TL, LS_A_TL2BR, 150 }, //# Fast arc top right to top left
|
|
{"TR2L Trans", BOTH_T1_TR__L, Q_TR, Q_L, AFLAG_ACTIVE, 100, BLK_NO, LS_R_R2L, LS_A_L2R, 150 }, //# Fast arc top right to left
|
|
{"TR2BL Trans", BOTH_T1_TR_BL, Q_TR, Q_BL, AFLAG_ACTIVE, 100, BLK_NO, LS_R_TR2BL, LS_A_BL2TR, 150 }, //# Fast weak spin top right to bottom left
|
|
{"T2BR Trans", BOTH_T1_T__BR, Q_T, Q_BR, AFLAG_ACTIVE, 100, BLK_NO, LS_R_TL2BR, LS_A_BR2TL, 150 }, //# Fast arc top to bottom right
|
|
{"T2R Trans", BOTH_T1_T___R, Q_T, Q_R, AFLAG_ACTIVE, 100, BLK_NO, LS_R_L2R, LS_A_R2L, 150 }, //# Fast arc top to right
|
|
{"T2TR Trans", BOTH_T1_T__TR, Q_T, Q_TR, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BL2TR, LS_A_TR2BL, 150 }, //# Fast arc top to top right
|
|
{"T2TL Trans", BOTH_T1_T__TL, Q_T, Q_TL, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BR2TL, LS_A_TL2BR, 150 }, //# Fast arc top to top left
|
|
{"T2L Trans", BOTH_T1_T___L, Q_T, Q_L, AFLAG_ACTIVE, 100, BLK_NO, LS_R_R2L, LS_A_L2R, 150 }, //# Fast arc top to left
|
|
{"T2BL Trans", BOTH_T1_T__BL, Q_T, Q_BL, AFLAG_ACTIVE, 100, BLK_NO, LS_R_TR2BL, LS_A_BL2TR, 150 }, //# Fast arc top to bottom left
|
|
{"TL2BR Trans", BOTH_T1_TL_BR, Q_TL, Q_BR, AFLAG_ACTIVE, 100, BLK_NO, LS_R_TL2BR, LS_A_BR2TL, 150 }, //# Fast weak spin top left to bottom right
|
|
{"TL2R Trans", BOTH_T1_TL__R, Q_TL, Q_R, AFLAG_ACTIVE, 100, BLK_NO, LS_R_L2R, LS_A_R2L, 150 }, //# Fast arc top left to right (use: BOTH_T1__R_TL)
|
|
{"TL2TR Trans", BOTH_T1_TL_TR, Q_TL, Q_TR, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BL2TR, LS_A_TR2BL, 150 }, //# Fast arc top left to top right (use: BOTH_T1_TR_TL)
|
|
{"TL2T Trans", BOTH_T1_TL_T_, Q_TL, Q_T, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BL2TR, LS_A_T2B, 150 }, //# Fast arc top left to top (use: BOTH_T1_T__TL)
|
|
{"TL2L Trans", BOTH_T1_TL__L, Q_TL, Q_L, AFLAG_ACTIVE, 100, BLK_NO, LS_R_R2L, LS_A_L2R, 150 }, //# Fast arc top left to left (use: BOTH_T1__L_TL)
|
|
{"TL2BL Trans", BOTH_T1_TL_BL, Q_TL, Q_BL, AFLAG_ACTIVE, 100, BLK_NO, LS_R_TR2BL, LS_A_BL2TR, 150 }, //# Fast arc top left to bottom left
|
|
{"L2BR Trans", BOTH_T1__L_BR, Q_L, Q_BR, AFLAG_ACTIVE, 100, BLK_NO, LS_R_TL2BR, LS_A_BR2TL, 150 }, //# Fast weak spin left to bottom right
|
|
{"L2R Trans", BOTH_T1__L__R, Q_L, Q_R, AFLAG_ACTIVE, 100, BLK_NO, LS_R_L2R, LS_A_R2L, 150 }, //# Fast weak spin left to right
|
|
{"L2TR Trans", BOTH_T1__L_TR, Q_L, Q_TR, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BL2TR, LS_A_TR2BL, 150 }, //# Fast arc left to top right (use: BOTH_T1_TR__L)
|
|
{"L2T Trans", BOTH_T1__L_T_, Q_L, Q_T, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BL2TR, LS_A_T2B, 150 }, //# Fast arc left to top (use: BOTH_T1_T___L)
|
|
{"L2TL Trans", BOTH_T1__L_TL, Q_L, Q_TL, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BR2TL, LS_A_TL2BR, 150 }, //# Fast arc left to top left
|
|
{"L2BL Trans", BOTH_T1__L_BL, Q_L, Q_BL, AFLAG_ACTIVE, 100, BLK_NO, LS_R_TR2BL, LS_A_BL2TR, 150 }, //# Fast arc left to bottom left (use: BOTH_T1_BL__L)
|
|
{"BL2BR Trans", BOTH_T1_BL_BR, Q_BL, Q_BR, AFLAG_ACTIVE, 100, BLK_NO, LS_R_TL2BR, LS_A_BR2TL, 150 }, //# Fast weak spin bottom left to bottom right
|
|
{"BL2R Trans", BOTH_T1_BL__R, Q_BL, Q_R, AFLAG_ACTIVE, 100, BLK_NO, LS_R_L2R, LS_A_R2L, 150 }, //# Fast weak spin bottom left to right
|
|
{"BL2TR Trans", BOTH_T1_BL_TR, Q_BL, Q_TR, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BL2TR, LS_A_TR2BL, 150 }, //# Fast weak spin bottom left to top right
|
|
{"BL2T Trans", BOTH_T1_BL_T_, Q_BL, Q_T, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BL2TR, LS_A_T2B, 150 }, //# Fast arc bottom left to top (use: BOTH_T1_T__BL)
|
|
{"BL2TL Trans", BOTH_T1_BL_TL, Q_BL, Q_TL, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BR2TL, LS_A_TL2BR, 150 }, //# Fast arc bottom left to top left (use: BOTH_T1_TL_BL)
|
|
{"BL2L Trans", BOTH_T1_BL__L, Q_BL, Q_L, AFLAG_ACTIVE, 100, BLK_NO, LS_R_R2L, LS_A_L2R, 150 }, //# Fast arc bottom left to left
|
|
|
|
//Bounces
|
|
{"Bounce BR", BOTH_B1_BR___, Q_BR, Q_BR, AFLAG_ACTIVE, 100, BLK_NO, LS_R_TL2BR, LS_T1_BR_TR, 150 },
|
|
{"Bounce R", BOTH_B1__R___, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_NO, LS_R_L2R, LS_T1__R__L, 150 },
|
|
{"Bounce TR", BOTH_B1_TR___, Q_TR, Q_TR, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BL2TR, LS_T1_TR_TL, 150 },
|
|
{"Bounce T", BOTH_B1_T____, Q_T, Q_T, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BL2TR, LS_T1_T__BL, 150 },
|
|
{"Bounce TL", BOTH_B1_TL___, Q_TL, Q_TL, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BR2TL, LS_T1_TL_TR, 150 },
|
|
{"Bounce L", BOTH_B1__L___, Q_L, Q_L, AFLAG_ACTIVE, 100, BLK_NO, LS_R_R2L, LS_T1__L__R, 150 },
|
|
{"Bounce BL", BOTH_B1_BL___, Q_BL, Q_BL, AFLAG_ACTIVE, 100, BLK_NO, LS_R_TR2BL, LS_T1_BL_TR, 150 },
|
|
|
|
//Deflected attacks (like bounces, but slide off enemy saber, not straight back)
|
|
{"Deflect BR", BOTH_D1_BR___, Q_BR, Q_BR, AFLAG_ACTIVE, 100, BLK_NO, LS_R_TL2BR, LS_T1_BR_TR, 150 },
|
|
{"Deflect R", BOTH_D1__R___, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_NO, LS_R_L2R, LS_T1__R__L, 150 },
|
|
{"Deflect TR", BOTH_D1_TR___, Q_TR, Q_TR, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BL2TR, LS_T1_TR_TL, 150 },
|
|
{"Deflect T", BOTH_B1_T____, Q_T, Q_T, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BL2TR, LS_T1_T__BL, 150 },
|
|
{"Deflect TL", BOTH_D1_TL___, Q_TL, Q_TL, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BR2TL, LS_T1_TL_TR, 150 },
|
|
{"Deflect L", BOTH_D1__L___, Q_L, Q_L, AFLAG_ACTIVE, 100, BLK_NO, LS_R_R2L, LS_T1__L__R, 150 },
|
|
{"Deflect BL", BOTH_D1_BL___, Q_BL, Q_BL, AFLAG_ACTIVE, 100, BLK_NO, LS_R_TR2BL, LS_T1_BL_TR, 150 },
|
|
{"Deflect B", BOTH_D1_B____, Q_B, Q_B, AFLAG_ACTIVE, 100, BLK_NO, LS_R_BL2TR, LS_T1_T__BL, 150 },
|
|
|
|
//Reflected attacks
|
|
{"Reflected BR",BOTH_V1_BR_S1, Q_BR, Q_BR, AFLAG_ACTIVE, 100, BLK_NO, LS_READY, LS_READY, 150 },// LS_V1_BR
|
|
{"Reflected R", BOTH_V1__R_S1, Q_R, Q_R, AFLAG_ACTIVE, 100, BLK_NO, LS_READY, LS_READY, 150 },// LS_V1__R
|
|
{"Reflected TR",BOTH_V1_TR_S1, Q_TR, Q_TR, AFLAG_ACTIVE, 100, BLK_NO, LS_READY, LS_READY, 150 },// LS_V1_TR
|
|
{"Reflected T", BOTH_V1_T__S1, Q_T, Q_T, AFLAG_ACTIVE, 100, BLK_NO, LS_READY, LS_READY, 150 },// LS_V1_T_
|
|
{"Reflected TL",BOTH_V1_TL_S1, Q_TL, Q_TL, AFLAG_ACTIVE, 100, BLK_NO, LS_READY, LS_READY, 150 },// LS_V1_TL
|
|
{"Reflected L", BOTH_V1__L_S1, Q_L, Q_L, AFLAG_ACTIVE, 100, BLK_NO, LS_READY, LS_READY, 150 },// LS_V1__L
|
|
{"Reflected BL",BOTH_V1_BL_S1, Q_BL, Q_BL, AFLAG_ACTIVE, 100, BLK_NO, LS_READY, LS_READY, 150 },// LS_V1_BL
|
|
{"Reflected B", BOTH_V1_B__S1, Q_B, Q_B, AFLAG_ACTIVE, 100, BLK_NO, LS_READY, LS_READY, 150 },// LS_V1_B_
|
|
|
|
// Broken parries
|
|
{"BParry Top", BOTH_H1_S1_T_, Q_T, Q_B, AFLAG_ACTIVE, 50, BLK_NO, LS_READY, LS_READY, 150 }, // LS_PARRY_UP,
|
|
{"BParry UR", BOTH_H1_S1_TR, Q_TR, Q_BL, AFLAG_ACTIVE, 50, BLK_NO, LS_READY, LS_READY, 150 }, // LS_PARRY_UR,
|
|
{"BParry UL", BOTH_H1_S1_TL, Q_TL, Q_BR, AFLAG_ACTIVE, 50, BLK_NO, LS_READY, LS_READY, 150 }, // LS_PARRY_UL,
|
|
{"BParry LR", BOTH_H1_S1_BL, Q_BL, Q_TR, AFLAG_ACTIVE, 50, BLK_NO, LS_READY, LS_READY, 150 }, // LS_PARRY_LR,
|
|
{"BParry Bot", BOTH_H1_S1_B_, Q_B, Q_T, AFLAG_ACTIVE, 50, BLK_NO, LS_READY, LS_READY, 150 }, // LS_PARRY_LL
|
|
{"BParry LL", BOTH_H1_S1_BR, Q_BR, Q_TL, AFLAG_ACTIVE, 50, BLK_NO, LS_READY, LS_READY, 150 }, // LS_PARRY_LL
|
|
|
|
// Knockaways
|
|
{"Knock Top", BOTH_K1_S1_T_, Q_R, Q_T, AFLAG_ACTIVE, 50, BLK_WIDE, LS_R_BL2TR, LS_T1_T__BR, 150 }, // LS_PARRY_UP,
|
|
{"Knock UR", BOTH_K1_S1_TR, Q_R, Q_TR, AFLAG_ACTIVE, 50, BLK_WIDE, LS_R_BL2TR, LS_T1_TR__R, 150 }, // LS_PARRY_UR,
|
|
{"Knock UL", BOTH_K1_S1_TL, Q_R, Q_TL, AFLAG_ACTIVE, 50, BLK_WIDE, LS_R_BR2TL, LS_T1_TL__L, 150 }, // LS_PARRY_UL,
|
|
{"Knock LR", BOTH_K1_S1_BL, Q_R, Q_BL, AFLAG_ACTIVE, 50, BLK_WIDE, LS_R_TL2BR, LS_T1_BL_TL, 150 }, // LS_PARRY_LR,
|
|
{"Knock LL", BOTH_K1_S1_BR, Q_R, Q_BR, AFLAG_ACTIVE, 50, BLK_WIDE, LS_R_TR2BL, LS_T1_BR_TR, 150 }, // LS_PARRY_LL
|
|
|
|
// Parry
|
|
{"Parry Top", BOTH_P1_S1_T_, Q_R, Q_T, AFLAG_ACTIVE, 50, BLK_WIDE, LS_R_BL2TR, LS_A_T2B, 150 }, // LS_PARRY_UP,
|
|
{"Parry UR", BOTH_P1_S1_TR, Q_R, Q_TL, AFLAG_ACTIVE, 50, BLK_WIDE, LS_R_BL2TR, LS_A_TR2BL, 150 }, // LS_PARRY_UR,
|
|
{"Parry UL", BOTH_P1_S1_TL, Q_R, Q_TR, AFLAG_ACTIVE, 50, BLK_WIDE, LS_R_BR2TL, LS_A_TL2BR, 150 }, // LS_PARRY_UL,
|
|
{"Parry LR", BOTH_P1_S1_BL, Q_R, Q_BR, AFLAG_ACTIVE, 50, BLK_WIDE, LS_R_TL2BR, LS_A_BR2TL, 150 }, // LS_PARRY_LR,
|
|
{"Parry LL", BOTH_P1_S1_BR, Q_R, Q_BL, AFLAG_ACTIVE, 50, BLK_WIDE, LS_R_TR2BL, LS_A_BL2TR, 150 }, // LS_PARRY_LL
|
|
|
|
// Reflecting a missile
|
|
{"Reflect Top", BOTH_P1_S1_T_, Q_R, Q_T, AFLAG_ACTIVE, 50, BLK_WIDE, LS_R_BL2TR, LS_A_T2B, 300 }, // LS_PARRY_UP,
|
|
{"Reflect UR", BOTH_P1_S1_TL, Q_R, Q_TR, AFLAG_ACTIVE, 50, BLK_WIDE, LS_R_BR2TL, LS_A_TL2BR, 300 }, // LS_PARRY_UR,
|
|
{"Reflect UL", BOTH_P1_S1_TR, Q_R, Q_TL, AFLAG_ACTIVE, 50, BLK_WIDE, LS_R_BL2TR, LS_A_TR2BL, 300 }, // LS_PARRY_UL,
|
|
{"Reflect LR", BOTH_P1_S1_BR, Q_R, Q_BL, AFLAG_ACTIVE, 50, BLK_WIDE, LS_R_TR2BL, LS_A_BL2TR, 300 }, // LS_PARRY_LR
|
|
{"Reflect LL", BOTH_P1_S1_BL, Q_R, Q_BR, AFLAG_ACTIVE, 50, BLK_WIDE, LS_R_TL2BR, LS_A_BR2TL, 300 }, // LS_PARRY_LL,
|
|
};
|
|
|
|
|
|
saberMoveName_t transitionMove[Q_NUM_QUADS][Q_NUM_QUADS] =
|
|
{
|
|
LS_NONE, //Can't transition to same pos!
|
|
LS_T1_BR__R,//40
|
|
LS_T1_BR_TR,
|
|
LS_T1_BR_T_,
|
|
LS_T1_BR_TL,
|
|
LS_T1_BR__L,
|
|
LS_T1_BR_BL,
|
|
LS_NONE, //No transitions to bottom, and no anims start there, so shouldn't need any
|
|
LS_T1__R_BR,//46
|
|
LS_NONE, //Can't transition to same pos!
|
|
LS_T1__R_TR,
|
|
LS_T1__R_T_,
|
|
LS_T1__R_TL,
|
|
LS_T1__R__L,
|
|
LS_T1__R_BL,
|
|
LS_NONE, //No transitions to bottom, and no anims start there, so shouldn't need any
|
|
LS_T1_TR_BR,//52
|
|
LS_T1_TR__R,
|
|
LS_NONE, //Can't transition to same pos!
|
|
LS_T1_TR_T_,
|
|
LS_T1_TR_TL,
|
|
LS_T1_TR__L,
|
|
LS_T1_TR_BL,
|
|
LS_NONE, //No transitions to bottom, and no anims start there, so shouldn't need any
|
|
LS_T1_T__BR,//58
|
|
LS_T1_T___R,
|
|
LS_T1_T__TR,
|
|
LS_NONE, //Can't transition to same pos!
|
|
LS_T1_T__TL,
|
|
LS_T1_T___L,
|
|
LS_T1_T__BL,
|
|
LS_NONE, //No transitions to bottom, and no anims start there, so shouldn't need any
|
|
LS_T1_TL_BR,//64
|
|
LS_T1_TL__R,
|
|
LS_T1_TL_TR,
|
|
LS_T1_TL_T_,
|
|
LS_NONE, //Can't transition to same pos!
|
|
LS_T1_TL__L,
|
|
LS_T1_TL_BL,
|
|
LS_NONE, //No transitions to bottom, and no anims start there, so shouldn't need any
|
|
LS_T1__L_BR,//70
|
|
LS_T1__L__R,
|
|
LS_T1__L_TR,
|
|
LS_T1__L_T_,
|
|
LS_T1__L_TL,
|
|
LS_NONE, //Can't transition to same pos!
|
|
LS_T1__L_BL,
|
|
LS_NONE, //No transitions to bottom, and no anims start there, so shouldn't need any
|
|
LS_T1_BL_BR,//76
|
|
LS_T1_BL__R,
|
|
LS_T1_BL_TR,
|
|
LS_T1_BL_T_,
|
|
LS_T1_BL_TL,
|
|
LS_T1_BL__L,
|
|
LS_NONE, //Can't transition to same pos!
|
|
LS_NONE, //No transitions to bottom, and no anims start there, so shouldn't need any
|
|
LS_T1_BL_BR,//NOTE: there are no transitions from bottom, so re-use the bottom right transitions
|
|
LS_T1_BR__R,
|
|
LS_T1_BR_TR,
|
|
LS_T1_BR_T_,
|
|
LS_T1_BR_TL,
|
|
LS_T1_BR__L,
|
|
LS_T1_BR_BL,
|
|
LS_NONE //No transitions to bottom, and no anims start there, so shouldn't need any
|
|
};
|
|
|
|
void PM_VelocityForSaberMove( playerState_t *ps, vec3_t throwDir )
|
|
{
|
|
vec3_t vForward, vRight, vUp, startQ, endQ;
|
|
|
|
AngleVectors( ps->viewangles, vForward, vRight, vUp );
|
|
|
|
switch ( saberMoveData[ps->saberMove].startQuad )
|
|
{
|
|
case Q_BR:
|
|
VectorScale( vRight, 1, startQ );
|
|
VectorMA( startQ, -1, vUp, startQ );
|
|
break;
|
|
case Q_R:
|
|
VectorScale( vRight, 2, startQ );
|
|
break;
|
|
case Q_TR:
|
|
VectorScale( vRight, 1, startQ );
|
|
VectorMA( startQ, 1, vUp, startQ );
|
|
break;
|
|
case Q_T:
|
|
VectorScale( vUp, 2, startQ );
|
|
break;
|
|
case Q_TL:
|
|
VectorScale( vRight, -1, startQ );
|
|
VectorMA( startQ, 1, vUp, startQ );
|
|
break;
|
|
case Q_L:
|
|
VectorScale( vRight, -2, startQ );
|
|
break;
|
|
case Q_BL:
|
|
VectorScale( vRight, -1, startQ );
|
|
VectorMA( startQ, -1, vUp, startQ );
|
|
break;
|
|
case Q_B:
|
|
VectorScale( vUp, -2, startQ );
|
|
break;
|
|
}
|
|
switch ( saberMoveData[ps->saberMove].endQuad )
|
|
{
|
|
case Q_BR:
|
|
VectorScale( vRight, 1, endQ );
|
|
VectorMA( endQ, -1, vUp, endQ );
|
|
break;
|
|
case Q_R:
|
|
VectorScale( vRight, 2, endQ );
|
|
break;
|
|
case Q_TR:
|
|
VectorScale( vRight, 1, endQ );
|
|
VectorMA( endQ, 1, vUp, endQ );
|
|
break;
|
|
case Q_T:
|
|
VectorScale( vUp, 2, endQ );
|
|
break;
|
|
case Q_TL:
|
|
VectorScale( vRight, -1, endQ );
|
|
VectorMA( endQ, 1, vUp, endQ );
|
|
break;
|
|
case Q_L:
|
|
VectorScale( vRight, -2, endQ );
|
|
break;
|
|
case Q_BL:
|
|
VectorScale( vRight, -1, endQ );
|
|
VectorMA( endQ, -1, vUp, endQ );
|
|
break;
|
|
case Q_B:
|
|
VectorScale( vUp, -2, endQ );
|
|
break;
|
|
}
|
|
VectorMA( endQ, 2, vForward, endQ );
|
|
VectorScale( throwDir, 125, throwDir );//FIXME: pass in the throw strength?
|
|
VectorSubtract( endQ, startQ, throwDir );
|
|
}
|
|
|
|
qboolean PM_VelocityForBlockedMove( playerState_t *ps, vec3_t throwDir )
|
|
{
|
|
vec3_t vForward, vRight, vUp;
|
|
AngleVectors( ps->viewangles, vForward, vRight, vUp );
|
|
switch ( ps->saberBlocked )
|
|
{
|
|
case BLOCKED_UPPER_RIGHT:
|
|
VectorScale( vRight, 1, throwDir );
|
|
VectorMA( throwDir, 1, vUp, throwDir );
|
|
break;
|
|
case BLOCKED_UPPER_LEFT:
|
|
VectorScale( vRight, -1, throwDir );
|
|
VectorMA( throwDir, 1, vUp, throwDir );
|
|
break;
|
|
case BLOCKED_LOWER_RIGHT:
|
|
VectorScale( vRight, 1, throwDir );
|
|
VectorMA( throwDir, -1, vUp, throwDir );
|
|
break;
|
|
case BLOCKED_LOWER_LEFT:
|
|
VectorScale( vRight, -1, throwDir );
|
|
VectorMA( throwDir, -1, vUp, throwDir );
|
|
break;
|
|
case BLOCKED_TOP:
|
|
VectorScale( vUp, 2, throwDir );
|
|
break;
|
|
default:
|
|
return qfalse;
|
|
break;
|
|
}
|
|
VectorMA( throwDir, 2, vForward, throwDir );
|
|
VectorScale( throwDir, 250, throwDir );//FIXME: pass in the throw strength?
|
|
return qtrue;
|
|
}
|
|
|
|
int PM_AnimLevelForSaberAnim( int anim )
|
|
{
|
|
if ( anim >= BOTH_A1_T__B_ && anim <= BOTH_D1_B____ )
|
|
{
|
|
return FORCE_LEVEL_1;
|
|
}
|
|
if ( anim >= BOTH_A2_T__B_ && anim <= BOTH_D2_B____ )
|
|
{
|
|
return FORCE_LEVEL_2;
|
|
}
|
|
if ( anim >= BOTH_A3_T__B_ && anim <= BOTH_D3_B____ )
|
|
{
|
|
return FORCE_LEVEL_3;
|
|
}
|
|
if ( anim >= BOTH_A4_T__B_ && anim <= BOTH_D4_B____ )
|
|
{//desann
|
|
return FORCE_LEVEL_4;
|
|
}
|
|
if ( anim >= BOTH_A5_T__B_ && anim <= BOTH_D5_B____ )
|
|
{//tavion
|
|
return FORCE_LEVEL_5;
|
|
}
|
|
if ( anim >= BOTH_A6_T__B_ && anim <= BOTH_D6_B____ )
|
|
{//dual
|
|
return SS_DUAL;
|
|
}
|
|
if ( anim >= BOTH_A7_T__B_ && anim <= BOTH_D7_B____ )
|
|
{//staff
|
|
return SS_STAFF;
|
|
}
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
|
|
int PM_PowerLevelForSaberAnim( playerState_t *ps, int saberNum )
|
|
{
|
|
int anim = ps->torsoAnim;
|
|
int animTimeElapsed = PM_AnimLength( g_entities[ps->clientNum].client->clientInfo.animFileIndex, (animNumber_t)anim ) - ps->torsoAnimTimer;
|
|
if ( anim >= BOTH_A1_T__B_ && anim <= BOTH_D1_B____ )
|
|
{
|
|
//FIXME: these two need their own style
|
|
if ( ps->saber[0].type == SABER_LANCE )
|
|
{
|
|
return FORCE_LEVEL_4;
|
|
}
|
|
else if ( ps->saber[0].type == SABER_TRIDENT )
|
|
{
|
|
return FORCE_LEVEL_3;
|
|
}
|
|
return FORCE_LEVEL_1;
|
|
}
|
|
if ( anim >= BOTH_A2_T__B_ && anim <= BOTH_D2_B____ )
|
|
{
|
|
return FORCE_LEVEL_2;
|
|
}
|
|
if ( anim >= BOTH_A3_T__B_ && anim <= BOTH_D3_B____ )
|
|
{
|
|
return FORCE_LEVEL_3;
|
|
}
|
|
if ( anim >= BOTH_A4_T__B_ && anim <= BOTH_D4_B____ )
|
|
{//desann
|
|
return FORCE_LEVEL_4;
|
|
}
|
|
if ( anim >= BOTH_A5_T__B_ && anim <= BOTH_D5_B____ )
|
|
{//tavion
|
|
return FORCE_LEVEL_2;
|
|
}
|
|
if ( anim >= BOTH_A6_T__B_ && anim <= BOTH_D6_B____ )
|
|
{//dual
|
|
return FORCE_LEVEL_2;
|
|
}
|
|
if ( anim >= BOTH_A7_T__B_ && anim <= BOTH_D7_B____ )
|
|
{//staff
|
|
return FORCE_LEVEL_2;
|
|
}
|
|
if ( ( anim >= BOTH_P1_S1_T_ && anim <= BOTH_P1_S1_BR )
|
|
|| ( anim >= BOTH_P6_S6_T_ && anim <= BOTH_P6_S6_BR )
|
|
|| ( anim >= BOTH_P7_S7_T_ && anim <= BOTH_P7_S7_BR ) )
|
|
{//parries
|
|
switch ( ps->saberAnimLevel )
|
|
{
|
|
case SS_STRONG:
|
|
case SS_DESANN:
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case SS_TAVION:
|
|
case SS_STAFF:
|
|
case SS_DUAL:
|
|
case SS_MEDIUM:
|
|
return FORCE_LEVEL_2;
|
|
break;
|
|
case SS_FAST:
|
|
return FORCE_LEVEL_1;
|
|
break;
|
|
default:
|
|
return FORCE_LEVEL_0;
|
|
break;
|
|
}
|
|
}
|
|
if ( ( anim >= BOTH_K1_S1_T_ && anim <= BOTH_K1_S1_BR )
|
|
|| ( anim >= BOTH_K6_S6_T_ && anim <= BOTH_K6_S6_BR )
|
|
|| ( anim >= BOTH_K7_S7_T_ && anim <= BOTH_K7_S7_BR ) )
|
|
{//knockaways
|
|
return FORCE_LEVEL_3;
|
|
}
|
|
if ( ( anim >= BOTH_V1_BR_S1 && anim <= BOTH_V1_B__S1 )
|
|
|| ( anim >= BOTH_V6_BR_S6 && anim <= BOTH_V6_B__S6 )
|
|
|| ( anim >= BOTH_V7_BR_S7 && anim <= BOTH_V7_B__S7 ) )
|
|
{//knocked-away attacks
|
|
return FORCE_LEVEL_1;
|
|
}
|
|
if ( ( anim >= BOTH_H1_S1_T_ && anim <= BOTH_H1_S1_BR )
|
|
|| ( anim >= BOTH_H6_S6_T_ && anim <= BOTH_H6_S6_BR )
|
|
|| ( anim >= BOTH_H7_S7_T_ && anim <= BOTH_H7_S7_BR ) )
|
|
{//broken parries
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
switch ( anim )
|
|
{
|
|
case BOTH_A2_STABBACK1:
|
|
if ( ps->torsoAnimTimer < 450 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else if ( animTimeElapsed < 400 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_ATTACK_BACK:
|
|
if ( ps->torsoAnimTimer < 500 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_CROUCHATTACKBACK1:
|
|
if ( ps->torsoAnimTimer < 800 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_BUTTERFLY_LEFT:
|
|
case BOTH_BUTTERFLY_RIGHT:
|
|
case BOTH_BUTTERFLY_FL1:
|
|
case BOTH_BUTTERFLY_FR1:
|
|
//FIXME: break up?
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_FJSS_TR_BL:
|
|
case BOTH_FJSS_TL_BR:
|
|
//FIXME: break up?
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_K1_S1_T_: //# knockaway saber top
|
|
case BOTH_K1_S1_TR: //# knockaway saber top right
|
|
case BOTH_K1_S1_TL: //# knockaway saber top left
|
|
case BOTH_K1_S1_BL: //# knockaway saber bottom left
|
|
case BOTH_K1_S1_B_: //# knockaway saber bottom
|
|
case BOTH_K1_S1_BR: //# knockaway saber bottom right
|
|
//FIXME: break up?
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_LUNGE2_B__T_:
|
|
if ( ps->torsoAnimTimer < 400 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else if ( animTimeElapsed < 150 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_FORCELEAP2_T__B_:
|
|
if ( ps->torsoAnimTimer < 400 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else if ( animTimeElapsed < 550 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_VS_ATR_S:
|
|
case BOTH_VS_ATL_S:
|
|
case BOTH_VT_ATR_S:
|
|
case BOTH_VT_ATL_S:
|
|
return FORCE_LEVEL_3;//???
|
|
break;
|
|
case BOTH_JUMPFLIPSLASHDOWN1:
|
|
if ( ps->torsoAnimTimer <= 900 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else if ( animTimeElapsed < 550 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_JUMPFLIPSTABDOWN:
|
|
if ( ps->torsoAnimTimer <= 1200 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else if ( animTimeElapsed <= 250 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_JUMPATTACK6:
|
|
/*
|
|
if (pm->ps)
|
|
{
|
|
if ( ( pm->ps->legsAnimTimer >= 1450
|
|
&& PM_AnimLength( g_entities[ps->clientNum].client->clientInfo.animFileIndex, BOTH_JUMPATTACK6 ) - pm->ps->legsAnimTimer >= 400 )
|
|
||(pm->ps->legsAnimTimer >= 400
|
|
&& PM_AnimLength( g_entities[ps->clientNum].client->clientInfo.animFileIndex, BOTH_JUMPATTACK6 ) - pm->ps->legsAnimTimer >= 1100 ) )
|
|
{//pretty much sideways
|
|
return FORCE_LEVEL_3;
|
|
}
|
|
}
|
|
*/
|
|
if ( ( ps->torsoAnimTimer >= 1450
|
|
&& animTimeElapsed >= 400 )
|
|
||(ps->torsoAnimTimer >= 400
|
|
&& animTimeElapsed >= 1100 ) )
|
|
{//pretty much sideways
|
|
return FORCE_LEVEL_3;
|
|
}
|
|
return FORCE_LEVEL_0;
|
|
break;
|
|
case BOTH_JUMPATTACK7:
|
|
if ( ps->torsoAnimTimer <= 1200 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else if ( animTimeElapsed < 200 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_SPINATTACK6:
|
|
if ( animTimeElapsed <= 200 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_SPINATTACK7:
|
|
if ( ps->torsoAnimTimer <= 500 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else if ( animTimeElapsed < 500 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_FORCELONGLEAP_ATTACK:
|
|
if ( animTimeElapsed <= 200 )
|
|
{//1st four frames of anim
|
|
return FORCE_LEVEL_3;
|
|
}
|
|
break;
|
|
/*
|
|
case BOTH_A7_KICK_F://these kicks attack, too
|
|
case BOTH_A7_KICK_B:
|
|
case BOTH_A7_KICK_R:
|
|
case BOTH_A7_KICK_L:
|
|
//FIXME: break up
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
*/
|
|
case BOTH_STABDOWN:
|
|
if ( ps->torsoAnimTimer <= 900 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_3;
|
|
}
|
|
break;
|
|
case BOTH_STABDOWN_STAFF:
|
|
if ( ps->torsoAnimTimer <= 850 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_3;
|
|
}
|
|
break;
|
|
case BOTH_STABDOWN_DUAL:
|
|
if ( ps->torsoAnimTimer <= 900 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_3;
|
|
}
|
|
break;
|
|
case BOTH_A6_SABERPROTECT:
|
|
if ( ps->torsoAnimTimer < 650 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_A7_SOULCAL:
|
|
if ( ps->torsoAnimTimer < 650 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else if ( animTimeElapsed < 600 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_A1_SPECIAL:
|
|
if ( ps->torsoAnimTimer < 600 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else if ( animTimeElapsed < 200 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_A2_SPECIAL:
|
|
if ( ps->torsoAnimTimer < 300 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else if ( animTimeElapsed < 200 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_A3_SPECIAL:
|
|
if ( ps->torsoAnimTimer < 700 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else if ( animTimeElapsed < 200 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_FLIP_ATTACK7:
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_PULL_IMPALE_STAB:
|
|
if ( ps->torsoAnimTimer < 1000 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_PULL_IMPALE_SWING:
|
|
if ( ps->torsoAnimTimer < 500 )//750 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else if ( animTimeElapsed < 650 )//600 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_ALORA_SPIN_SLASH:
|
|
if ( ps->torsoAnimTimer < 900 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else if ( animTimeElapsed < 250 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_A6_FB:
|
|
if ( ps->torsoAnimTimer < 250 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else if ( animTimeElapsed < 250 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_A6_LR:
|
|
if ( ps->torsoAnimTimer < 250 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else if ( animTimeElapsed < 250 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_3;
|
|
break;
|
|
case BOTH_A7_HILT:
|
|
return FORCE_LEVEL_0;
|
|
break;
|
|
//===SABERLOCK SUPERBREAKS START===========================================================================
|
|
case BOTH_LK_S_DL_T_SB_1_W:
|
|
if ( ps->torsoAnimTimer < 700 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_5;
|
|
break;
|
|
case BOTH_LK_S_ST_S_SB_1_W:
|
|
if ( ps->torsoAnimTimer < 300 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_5;
|
|
break;
|
|
case BOTH_LK_S_DL_S_SB_1_W:
|
|
case BOTH_LK_S_S_S_SB_1_W:
|
|
if ( ps->torsoAnimTimer < 700 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else if ( animTimeElapsed < 400 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_5;
|
|
break;
|
|
case BOTH_LK_S_ST_T_SB_1_W:
|
|
case BOTH_LK_S_S_T_SB_1_W:
|
|
if ( ps->torsoAnimTimer < 150 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else if ( animTimeElapsed < 400 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_5;
|
|
break;
|
|
case BOTH_LK_DL_DL_T_SB_1_W:
|
|
return FORCE_LEVEL_5;
|
|
break;
|
|
case BOTH_LK_DL_DL_S_SB_1_W:
|
|
case BOTH_LK_DL_ST_S_SB_1_W:
|
|
if ( animTimeElapsed < 1000 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_5;
|
|
break;
|
|
case BOTH_LK_DL_ST_T_SB_1_W:
|
|
if ( ps->torsoAnimTimer < 950 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else if ( animTimeElapsed < 650 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_5;
|
|
break;
|
|
case BOTH_LK_DL_S_S_SB_1_W:
|
|
if ( saberNum != 0 )
|
|
{//only right hand saber does damage in this suberbreak
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
if ( ps->torsoAnimTimer < 900 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else if ( animTimeElapsed < 450 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_5;
|
|
break;
|
|
case BOTH_LK_DL_S_T_SB_1_W:
|
|
if ( saberNum != 0 )
|
|
{//only right hand saber does damage in this suberbreak
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
if ( ps->torsoAnimTimer < 250 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else if ( animTimeElapsed < 150 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_5;
|
|
break;
|
|
case BOTH_LK_ST_DL_S_SB_1_W:
|
|
return FORCE_LEVEL_5;
|
|
break;
|
|
case BOTH_LK_ST_DL_T_SB_1_W:
|
|
//special suberbreak - doesn't kill, just kicks them backwards
|
|
return FORCE_LEVEL_0;
|
|
break;
|
|
case BOTH_LK_ST_ST_S_SB_1_W:
|
|
case BOTH_LK_ST_S_S_SB_1_W:
|
|
if ( ps->torsoAnimTimer < 800 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else if ( animTimeElapsed < 350 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
return FORCE_LEVEL_5;
|
|
break;
|
|
case BOTH_LK_ST_ST_T_SB_1_W:
|
|
case BOTH_LK_ST_S_T_SB_1_W:
|
|
return FORCE_LEVEL_5;
|
|
break;
|
|
//===SABERLOCK SUPERBREAKS START===========================================================================
|
|
case BOTH_HANG_ATTACK:
|
|
//FIME: break up
|
|
if ( ps->torsoAnimTimer < 1000 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else if ( animTimeElapsed < 250 )
|
|
{//beginning of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else
|
|
{//sweet spot
|
|
return FORCE_LEVEL_5;
|
|
}
|
|
break;
|
|
case BOTH_ROLL_STAB:
|
|
if ( animTimeElapsed > 400 )
|
|
{//end of anim
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
else
|
|
{
|
|
return FORCE_LEVEL_3;
|
|
}
|
|
break;
|
|
}
|
|
return FORCE_LEVEL_0;
|
|
}
|
|
|
|
qboolean PM_InAnimForSaberMove( int anim, int saberMove )
|
|
{
|
|
switch ( anim )
|
|
{//special case anims
|
|
case BOTH_A2_STABBACK1:
|
|
case BOTH_ATTACK_BACK:
|
|
case BOTH_CROUCHATTACKBACK1:
|
|
case BOTH_ROLL_STAB:
|
|
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_JUMPFLIPSLASHDOWN1://#
|
|
case BOTH_JUMPFLIPSTABDOWN://#
|
|
case BOTH_JUMPATTACK6:
|
|
case BOTH_JUMPATTACK7:
|
|
case BOTH_SPINATTACK6:
|
|
case BOTH_SPINATTACK7:
|
|
case BOTH_VS_ATR_S:
|
|
case BOTH_VS_ATL_S:
|
|
case BOTH_VT_ATR_S:
|
|
case BOTH_VT_ATL_S:
|
|
case BOTH_FORCELONGLEAP_ATTACK:
|
|
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_FLIP_ATTACK7:
|
|
case BOTH_PULL_IMPALE_STAB:
|
|
case BOTH_PULL_IMPALE_SWING:
|
|
case BOTH_ALORA_SPIN_SLASH:
|
|
case BOTH_A6_FB:
|
|
case BOTH_A6_LR:
|
|
case BOTH_A7_HILT:
|
|
case BOTH_LK_S_DL_S_SB_1_W:
|
|
case BOTH_LK_S_DL_T_SB_1_W:
|
|
case BOTH_LK_S_ST_S_SB_1_W:
|
|
case BOTH_LK_S_ST_T_SB_1_W:
|
|
case BOTH_LK_S_S_S_SB_1_W:
|
|
case BOTH_LK_S_S_T_SB_1_W:
|
|
case BOTH_LK_DL_DL_S_SB_1_W:
|
|
case BOTH_LK_DL_DL_T_SB_1_W:
|
|
case BOTH_LK_DL_ST_S_SB_1_W:
|
|
case BOTH_LK_DL_ST_T_SB_1_W:
|
|
case BOTH_LK_DL_S_S_SB_1_W:
|
|
case BOTH_LK_DL_S_T_SB_1_W:
|
|
case BOTH_LK_ST_DL_S_SB_1_W:
|
|
case BOTH_LK_ST_DL_T_SB_1_W:
|
|
case BOTH_LK_ST_ST_S_SB_1_W:
|
|
case BOTH_LK_ST_ST_T_SB_1_W:
|
|
case BOTH_LK_ST_S_S_SB_1_W:
|
|
case BOTH_LK_ST_S_T_SB_1_W:
|
|
case BOTH_HANG_ATTACK:
|
|
return qtrue;
|
|
}
|
|
if ( PM_SaberDrawPutawayAnim( anim ) )
|
|
{
|
|
if ( saberMove == LS_DRAW || saberMove == LS_PUTAWAY )
|
|
{
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
else if ( PM_SaberStanceAnim( anim ) )
|
|
{
|
|
if ( saberMove == LS_READY )
|
|
{
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
int animLevel = PM_AnimLevelForSaberAnim( anim );
|
|
if ( animLevel <= 0 )
|
|
{//NOTE: this will always return false for the ready poses and putaway/draw...
|
|
return qfalse;
|
|
}
|
|
//drop the anim to the first level and start the checks there
|
|
anim -= (animLevel-FORCE_LEVEL_1)*SABER_ANIM_GROUP_SIZE;
|
|
//check level 1
|
|
if ( anim == saberMoveData[saberMove].animToUse )
|
|
{
|
|
return qtrue;
|
|
}
|
|
//check level 2
|
|
anim += SABER_ANIM_GROUP_SIZE;
|
|
if ( anim == saberMoveData[saberMove].animToUse )
|
|
{
|
|
return qtrue;
|
|
}
|
|
//check level 3
|
|
anim += SABER_ANIM_GROUP_SIZE;
|
|
if ( anim == saberMoveData[saberMove].animToUse )
|
|
{
|
|
return qtrue;
|
|
}
|
|
//check level 4
|
|
anim += SABER_ANIM_GROUP_SIZE;
|
|
if ( anim == saberMoveData[saberMove].animToUse )
|
|
{
|
|
return qtrue;
|
|
}
|
|
//check level 5
|
|
anim += SABER_ANIM_GROUP_SIZE;
|
|
if ( anim == saberMoveData[saberMove].animToUse )
|
|
{
|
|
return qtrue;
|
|
}
|
|
if ( anim >= BOTH_P1_S1_T_ && anim <= BOTH_H1_S1_BR )
|
|
{//parries, knockaways and broken parries
|
|
return (anim==saberMoveData[saberMove].animToUse);
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean PM_SaberInIdle( int move )
|
|
{
|
|
switch ( move )
|
|
{
|
|
case LS_NONE:
|
|
case LS_READY:
|
|
case LS_DRAW:
|
|
case LS_PUTAWAY:
|
|
return qtrue;
|
|
break;
|
|
}
|
|
return qfalse;
|
|
}
|
|
qboolean PM_SaberInSpecialAttack( int anim )
|
|
{
|
|
switch ( anim )
|
|
{
|
|
case BOTH_A2_STABBACK1:
|
|
case BOTH_ATTACK_BACK:
|
|
case BOTH_CROUCHATTACKBACK1:
|
|
case BOTH_ROLL_STAB:
|
|
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_JUMPFLIPSLASHDOWN1://#
|
|
case BOTH_JUMPFLIPSTABDOWN://#
|
|
case BOTH_JUMPATTACK6:
|
|
case BOTH_JUMPATTACK7:
|
|
case BOTH_SPINATTACK6:
|
|
case BOTH_SPINATTACK7:
|
|
case BOTH_FORCELONGLEAP_ATTACK:
|
|
case BOTH_VS_ATR_S:
|
|
case BOTH_VS_ATL_S:
|
|
case BOTH_VT_ATR_S:
|
|
case BOTH_VT_ATL_S:
|
|
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_FLIP_ATTACK7:
|
|
case BOTH_PULL_IMPALE_STAB:
|
|
case BOTH_PULL_IMPALE_SWING:
|
|
case BOTH_ALORA_SPIN_SLASH:
|
|
case BOTH_A6_FB:
|
|
case BOTH_A6_LR:
|
|
case BOTH_A7_HILT:
|
|
case BOTH_LK_S_DL_S_SB_1_W:
|
|
case BOTH_LK_S_DL_T_SB_1_W:
|
|
case BOTH_LK_S_ST_S_SB_1_W:
|
|
case BOTH_LK_S_ST_T_SB_1_W:
|
|
case BOTH_LK_S_S_S_SB_1_W:
|
|
case BOTH_LK_S_S_T_SB_1_W:
|
|
case BOTH_LK_DL_DL_S_SB_1_W:
|
|
case BOTH_LK_DL_DL_T_SB_1_W:
|
|
case BOTH_LK_DL_ST_S_SB_1_W:
|
|
case BOTH_LK_DL_ST_T_SB_1_W:
|
|
case BOTH_LK_DL_S_S_SB_1_W:
|
|
case BOTH_LK_DL_S_T_SB_1_W:
|
|
case BOTH_LK_ST_DL_S_SB_1_W:
|
|
case BOTH_LK_ST_DL_T_SB_1_W:
|
|
case BOTH_LK_ST_ST_S_SB_1_W:
|
|
case BOTH_LK_ST_ST_T_SB_1_W:
|
|
case BOTH_LK_ST_S_S_SB_1_W:
|
|
case BOTH_LK_ST_S_T_SB_1_W:
|
|
case BOTH_HANG_ATTACK:
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean PM_SaberInAttackPure( int move )
|
|
{
|
|
if ( move >= LS_A_TL2BR && move <= LS_A_T2B )
|
|
{
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
qboolean PM_SaberInAttack( int move )
|
|
{
|
|
if ( move >= LS_A_TL2BR && move <= LS_A_T2B )
|
|
{
|
|
return qtrue;
|
|
}
|
|
switch ( move )
|
|
{
|
|
case LS_A_BACK:
|
|
case LS_A_BACK_CR:
|
|
case LS_A_BACKSTAB:
|
|
case LS_ROLL_STAB:
|
|
case LS_A_LUNGE:
|
|
case LS_A_JUMP_T__B_:
|
|
case LS_A_FLIP_STAB:
|
|
case LS_A_FLIP_SLASH:
|
|
case LS_JUMPATTACK_DUAL:
|
|
case LS_JUMPATTACK_ARIAL_LEFT:
|
|
case LS_JUMPATTACK_ARIAL_RIGHT:
|
|
case LS_JUMPATTACK_CART_LEFT:
|
|
case LS_JUMPATTACK_CART_RIGHT:
|
|
case LS_JUMPATTACK_STAFF_LEFT:
|
|
case LS_JUMPATTACK_STAFF_RIGHT:
|
|
case LS_BUTTERFLY_LEFT:
|
|
case LS_BUTTERFLY_RIGHT:
|
|
case LS_A_BACKFLIP_ATK:
|
|
case LS_SPINATTACK_DUAL:
|
|
case LS_SPINATTACK:
|
|
case LS_LEAP_ATTACK:
|
|
case LS_SWOOP_ATTACK_RIGHT:
|
|
case LS_SWOOP_ATTACK_LEFT:
|
|
case LS_TAUNTAUN_ATTACK_RIGHT:
|
|
case LS_TAUNTAUN_ATTACK_LEFT:
|
|
case LS_KICK_F:
|
|
case LS_KICK_B:
|
|
case LS_KICK_R:
|
|
case LS_KICK_L:
|
|
case LS_KICK_S:
|
|
case LS_KICK_BF:
|
|
case LS_KICK_RL:
|
|
case LS_KICK_F_AIR:
|
|
case LS_KICK_B_AIR:
|
|
case LS_KICK_R_AIR:
|
|
case LS_KICK_L_AIR:
|
|
case LS_STABDOWN:
|
|
case LS_STABDOWN_STAFF:
|
|
case LS_STABDOWN_DUAL:
|
|
case LS_DUAL_SPIN_PROTECT:
|
|
case LS_STAFF_SOULCAL:
|
|
case LS_A1_SPECIAL:
|
|
case LS_A2_SPECIAL:
|
|
case LS_A3_SPECIAL:
|
|
case LS_UPSIDE_DOWN_ATTACK:
|
|
case LS_PULL_ATTACK_STAB:
|
|
case LS_PULL_ATTACK_SWING:
|
|
case LS_SPINATTACK_ALORA:
|
|
case LS_DUAL_FB:
|
|
case LS_DUAL_LR:
|
|
case LS_HILT_BASH:
|
|
return qtrue;
|
|
break;
|
|
}
|
|
return qfalse;
|
|
}
|
|
qboolean PM_SaberInTransition( int move )
|
|
{
|
|
if ( move >= LS_T1_BR__R && move <= LS_T1_BL__L )
|
|
{
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
qboolean PM_SaberInStart( int move )
|
|
{
|
|
if ( move >= LS_S_TL2BR && move <= LS_S_T2B )
|
|
{
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
qboolean PM_SaberInReturn( int move )
|
|
{
|
|
if ( move >= LS_R_TL2BR && move <= LS_R_T2B )
|
|
{
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
qboolean PM_SaberInTransitionAny( int move )
|
|
{
|
|
if ( PM_SaberInStart( move ) )
|
|
{
|
|
return qtrue;
|
|
}
|
|
else if ( PM_SaberInTransition( move ) )
|
|
{
|
|
return qtrue;
|
|
}
|
|
else if ( PM_SaberInReturn( move ) )
|
|
{
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
qboolean PM_SaberInBounce( int move )
|
|
{
|
|
if ( move >= LS_B1_BR && move <= LS_B1_BL )
|
|
{
|
|
return qtrue;
|
|
}
|
|
if ( move >= LS_D1_BR && move <= LS_D1_BL )
|
|
{
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
qboolean PM_SaberInBrokenParry( int move )
|
|
{
|
|
if ( move >= LS_V1_BR && move <= LS_V1_B_ )
|
|
{
|
|
return qtrue;
|
|
}
|
|
if ( move >= LS_H1_T_ && move <= LS_H1_BL )
|
|
{
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
qboolean PM_SaberInDeflect( int move )
|
|
{
|
|
if ( move >= LS_D1_BR && move <= LS_D1_B_ )
|
|
{
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
qboolean PM_SaberInParry( int move )
|
|
{
|
|
if ( move >= LS_PARRY_UP && move <= LS_PARRY_LL )
|
|
{
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
qboolean PM_SaberInKnockaway( int move )
|
|
{
|
|
if ( move >= LS_K1_T_ && move <= LS_K1_BL )
|
|
{
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
qboolean PM_SaberInReflect( int move )
|
|
{
|
|
if ( move >= LS_REFLECT_UP && move <= LS_REFLECT_LL )
|
|
{
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean PM_SaberInSpecial( int move )
|
|
{
|
|
switch( move )
|
|
{
|
|
case LS_A_BACK:
|
|
case LS_A_BACK_CR:
|
|
case LS_A_BACKSTAB:
|
|
case LS_ROLL_STAB:
|
|
case LS_A_LUNGE:
|
|
case LS_A_JUMP_T__B_:
|
|
case LS_A_FLIP_STAB:
|
|
case LS_A_FLIP_SLASH:
|
|
case LS_JUMPATTACK_DUAL:
|
|
case LS_JUMPATTACK_ARIAL_LEFT:
|
|
case LS_JUMPATTACK_ARIAL_RIGHT:
|
|
case LS_JUMPATTACK_CART_LEFT:
|
|
case LS_JUMPATTACK_CART_RIGHT:
|
|
case LS_JUMPATTACK_STAFF_LEFT:
|
|
case LS_JUMPATTACK_STAFF_RIGHT:
|
|
case LS_BUTTERFLY_LEFT:
|
|
case LS_BUTTERFLY_RIGHT:
|
|
case LS_A_BACKFLIP_ATK:
|
|
case LS_SPINATTACK_DUAL:
|
|
case LS_SPINATTACK:
|
|
case LS_LEAP_ATTACK:
|
|
case LS_SWOOP_ATTACK_RIGHT:
|
|
case LS_SWOOP_ATTACK_LEFT:
|
|
case LS_TAUNTAUN_ATTACK_RIGHT:
|
|
case LS_TAUNTAUN_ATTACK_LEFT:
|
|
case LS_KICK_F:
|
|
case LS_KICK_B:
|
|
case LS_KICK_R:
|
|
case LS_KICK_L:
|
|
case LS_KICK_S:
|
|
case LS_KICK_BF:
|
|
case LS_KICK_RL:
|
|
case LS_KICK_F_AIR:
|
|
case LS_KICK_B_AIR:
|
|
case LS_KICK_R_AIR:
|
|
case LS_KICK_L_AIR:
|
|
case LS_STABDOWN:
|
|
case LS_STABDOWN_STAFF:
|
|
case LS_STABDOWN_DUAL:
|
|
case LS_DUAL_SPIN_PROTECT:
|
|
case LS_STAFF_SOULCAL:
|
|
case LS_A1_SPECIAL:
|
|
case LS_A2_SPECIAL:
|
|
case LS_A3_SPECIAL:
|
|
case LS_UPSIDE_DOWN_ATTACK:
|
|
case LS_PULL_ATTACK_STAB:
|
|
case LS_PULL_ATTACK_SWING:
|
|
case LS_SPINATTACK_ALORA:
|
|
case LS_DUAL_FB:
|
|
case LS_DUAL_LR:
|
|
case LS_HILT_BASH:
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean PM_KickMove( int move )
|
|
{
|
|
switch( move )
|
|
{
|
|
case LS_KICK_F:
|
|
case LS_KICK_B:
|
|
case LS_KICK_R:
|
|
case LS_KICK_L:
|
|
case LS_KICK_S:
|
|
case LS_KICK_BF:
|
|
case LS_KICK_RL:
|
|
case LS_HILT_BASH:
|
|
case LS_KICK_F_AIR:
|
|
case LS_KICK_B_AIR:
|
|
case LS_KICK_R_AIR:
|
|
case LS_KICK_L_AIR:
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean PM_SaberCanInterruptMove( int move, int anim )
|
|
{
|
|
if ( PM_InAnimForSaberMove( anim, move ) )
|
|
{
|
|
switch( move )
|
|
{
|
|
case LS_A_BACK:
|
|
case LS_A_BACK_CR:
|
|
case LS_A_BACKSTAB:
|
|
case LS_ROLL_STAB:
|
|
case LS_A_LUNGE:
|
|
case LS_A_JUMP_T__B_:
|
|
case LS_A_FLIP_STAB:
|
|
case LS_A_FLIP_SLASH:
|
|
case LS_JUMPATTACK_DUAL:
|
|
case LS_JUMPATTACK_CART_LEFT:
|
|
case LS_JUMPATTACK_CART_RIGHT:
|
|
case LS_JUMPATTACK_STAFF_LEFT:
|
|
case LS_JUMPATTACK_STAFF_RIGHT:
|
|
case LS_BUTTERFLY_LEFT:
|
|
case LS_BUTTERFLY_RIGHT:
|
|
case LS_A_BACKFLIP_ATK:
|
|
case LS_SPINATTACK_DUAL:
|
|
case LS_SPINATTACK:
|
|
case LS_LEAP_ATTACK:
|
|
case LS_SWOOP_ATTACK_RIGHT:
|
|
case LS_SWOOP_ATTACK_LEFT:
|
|
case LS_TAUNTAUN_ATTACK_RIGHT:
|
|
case LS_TAUNTAUN_ATTACK_LEFT:
|
|
case LS_KICK_S:
|
|
case LS_KICK_BF:
|
|
case LS_KICK_RL:
|
|
case LS_STABDOWN:
|
|
case LS_STABDOWN_STAFF:
|
|
case LS_STABDOWN_DUAL:
|
|
case LS_DUAL_SPIN_PROTECT:
|
|
case LS_STAFF_SOULCAL:
|
|
case LS_A1_SPECIAL:
|
|
case LS_A2_SPECIAL:
|
|
case LS_A3_SPECIAL:
|
|
case LS_UPSIDE_DOWN_ATTACK:
|
|
case LS_PULL_ATTACK_STAB:
|
|
case LS_PULL_ATTACK_SWING:
|
|
case LS_SPINATTACK_ALORA:
|
|
case LS_DUAL_FB:
|
|
case LS_DUAL_LR:
|
|
case LS_HILT_BASH:
|
|
return qfalse;
|
|
}
|
|
|
|
if ( PM_SaberInAttackPure( move ) )
|
|
{
|
|
return qfalse;
|
|
}
|
|
if ( PM_SaberInStart( move ) )
|
|
{
|
|
return qfalse;
|
|
}
|
|
if ( PM_SaberInTransition( move ) )
|
|
{
|
|
return qfalse;
|
|
}
|
|
if ( PM_SaberInBounce( move ) )
|
|
{
|
|
return qfalse;
|
|
}
|
|
if ( PM_SaberInBrokenParry( move ) )
|
|
{
|
|
return qfalse;
|
|
}
|
|
if ( PM_SaberInDeflect( move ) )
|
|
{
|
|
return qfalse;
|
|
}
|
|
if ( PM_SaberInParry( move ) )
|
|
{
|
|
return qfalse;
|
|
}
|
|
if ( PM_SaberInKnockaway( move ) )
|
|
{
|
|
return qfalse;
|
|
}
|
|
if ( PM_SaberInReflect( move ) )
|
|
{
|
|
return qfalse;
|
|
}
|
|
}
|
|
switch ( anim )
|
|
{
|
|
case BOTH_A2_STABBACK1:
|
|
case BOTH_ATTACK_BACK:
|
|
case BOTH_CROUCHATTACKBACK1:
|
|
case BOTH_ROLL_STAB:
|
|
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_JUMPFLIPSLASHDOWN1://#
|
|
case BOTH_JUMPFLIPSTABDOWN://#
|
|
case BOTH_JUMPATTACK6:
|
|
case BOTH_JUMPATTACK7:
|
|
case BOTH_SPINATTACK6:
|
|
case BOTH_SPINATTACK7:
|
|
case BOTH_FORCELONGLEAP_ATTACK:
|
|
case BOTH_VS_ATR_S:
|
|
case BOTH_VS_ATL_S:
|
|
case BOTH_VT_ATR_S:
|
|
case BOTH_VT_ATL_S:
|
|
case BOTH_A7_KICK_S:
|
|
case BOTH_A7_KICK_BF:
|
|
case BOTH_A7_KICK_RL:
|
|
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_FLIP_ATTACK7:
|
|
case BOTH_PULL_IMPALE_STAB:
|
|
case BOTH_PULL_IMPALE_SWING:
|
|
case BOTH_ALORA_SPIN_SLASH:
|
|
case BOTH_A6_FB:
|
|
case BOTH_A6_LR:
|
|
case BOTH_A7_HILT:
|
|
case BOTH_LK_S_DL_S_SB_1_W:
|
|
case BOTH_LK_S_DL_T_SB_1_W:
|
|
case BOTH_LK_S_ST_S_SB_1_W:
|
|
case BOTH_LK_S_ST_T_SB_1_W:
|
|
case BOTH_LK_S_S_S_SB_1_W:
|
|
case BOTH_LK_S_S_T_SB_1_W:
|
|
case BOTH_LK_DL_DL_S_SB_1_W:
|
|
case BOTH_LK_DL_DL_T_SB_1_W:
|
|
case BOTH_LK_DL_ST_S_SB_1_W:
|
|
case BOTH_LK_DL_ST_T_SB_1_W:
|
|
case BOTH_LK_DL_S_S_SB_1_W:
|
|
case BOTH_LK_DL_S_T_SB_1_W:
|
|
case BOTH_LK_ST_DL_S_SB_1_W:
|
|
case BOTH_LK_ST_DL_T_SB_1_W:
|
|
case BOTH_LK_ST_ST_S_SB_1_W:
|
|
case BOTH_LK_ST_ST_T_SB_1_W:
|
|
case BOTH_LK_ST_S_S_SB_1_W:
|
|
case BOTH_LK_ST_S_T_SB_1_W:
|
|
case BOTH_HANG_ATTACK:
|
|
return qfalse;
|
|
}
|
|
return qtrue;
|
|
}
|
|
|
|
saberMoveName_t PM_BrokenParryForAttack( int move )
|
|
{
|
|
//Our attack was knocked away by a knockaway parry
|
|
//FIXME: need actual anims for this
|
|
//FIXME: need to know which side of the saber was hit! For now, we presume the saber gets knocked away from the center
|
|
switch ( saberMoveData[move].startQuad )
|
|
{
|
|
case Q_B:
|
|
return LS_V1_B_;
|
|
break;
|
|
case Q_BR:
|
|
return LS_V1_BR;
|
|
break;
|
|
case Q_R:
|
|
return LS_V1__R;
|
|
break;
|
|
case Q_TR:
|
|
return LS_V1_TR;
|
|
break;
|
|
case Q_T:
|
|
return LS_V1_T_;
|
|
break;
|
|
case Q_TL:
|
|
return LS_V1_TL;
|
|
break;
|
|
case Q_L:
|
|
return LS_V1__L;
|
|
break;
|
|
case Q_BL:
|
|
return LS_V1_BL;
|
|
break;
|
|
}
|
|
return LS_NONE;
|
|
}
|
|
|
|
saberMoveName_t PM_BrokenParryForParry( int move )
|
|
{
|
|
//FIXME: need actual anims for this
|
|
//FIXME: need to know which side of the saber was hit! For now, we presume the saber gets knocked away from the center
|
|
switch ( move )
|
|
{
|
|
case LS_PARRY_UP:
|
|
//Hmm... since we don't know what dir the hit came from, randomly pick knock down or knock back
|
|
if ( Q_irand( 0, 1 ) )
|
|
{
|
|
return LS_H1_B_;
|
|
}
|
|
else
|
|
{
|
|
return LS_H1_T_;
|
|
}
|
|
break;
|
|
case LS_PARRY_UR:
|
|
return LS_H1_TR;
|
|
break;
|
|
case LS_PARRY_UL:
|
|
return LS_H1_TL;
|
|
break;
|
|
case LS_PARRY_LR:
|
|
return LS_H1_BR;
|
|
break;
|
|
case LS_PARRY_LL:
|
|
return LS_H1_BL;
|
|
break;
|
|
case LS_READY:
|
|
return LS_H1_B_;//???
|
|
break;
|
|
}
|
|
return LS_NONE;
|
|
}
|
|
|
|
saberMoveName_t PM_KnockawayForParry( int move )
|
|
{
|
|
//FIXME: need actual anims for this
|
|
//FIXME: need to know which side of the saber was hit! For now, we presume the saber gets knocked away from the center
|
|
switch ( move )
|
|
{
|
|
case BLOCKED_TOP://LS_PARRY_UP:
|
|
return LS_K1_T_;//push up
|
|
break;
|
|
case BLOCKED_UPPER_RIGHT://LS_PARRY_UR:
|
|
default://case LS_READY:
|
|
return LS_K1_TR;//push up, slightly to right
|
|
break;
|
|
case BLOCKED_UPPER_LEFT://LS_PARRY_UL:
|
|
return LS_K1_TL;//push up and to left
|
|
break;
|
|
case BLOCKED_LOWER_RIGHT://LS_PARRY_LR:
|
|
return LS_K1_BR;//push down and to left
|
|
break;
|
|
case BLOCKED_LOWER_LEFT://LS_PARRY_LL:
|
|
return LS_K1_BL;//push down and to right
|
|
break;
|
|
}
|
|
//return LS_NONE;
|
|
}
|
|
|
|
saberMoveName_t PM_SaberBounceForAttack( int move )
|
|
{
|
|
switch ( saberMoveData[move].startQuad )
|
|
{
|
|
case Q_B:
|
|
case Q_BR:
|
|
return LS_B1_BR;
|
|
break;
|
|
case Q_R:
|
|
return LS_B1__R;
|
|
break;
|
|
case Q_TR:
|
|
return LS_B1_TR;
|
|
break;
|
|
case Q_T:
|
|
return LS_B1_T_;
|
|
break;
|
|
case Q_TL:
|
|
return LS_B1_TL;
|
|
break;
|
|
case Q_L:
|
|
return LS_B1__L;
|
|
break;
|
|
case Q_BL:
|
|
return LS_B1_BL;
|
|
break;
|
|
}
|
|
return LS_NONE;
|
|
}
|
|
|
|
saberMoveName_t PM_AttackMoveForQuad( int quad )
|
|
{
|
|
switch ( quad )
|
|
{
|
|
case Q_B:
|
|
case Q_BR:
|
|
return LS_A_BR2TL;
|
|
break;
|
|
case Q_R:
|
|
return LS_A_R2L;
|
|
break;
|
|
case Q_TR:
|
|
return LS_A_TR2BL;
|
|
break;
|
|
case Q_T:
|
|
return LS_A_T2B;
|
|
break;
|
|
case Q_TL:
|
|
return LS_A_TL2BR;
|
|
break;
|
|
case Q_L:
|
|
return LS_A_L2R;
|
|
break;
|
|
case Q_BL:
|
|
return LS_A_BL2TR;
|
|
break;
|
|
}
|
|
return LS_NONE;
|
|
}
|
|
|
|
int saberMoveTransitionAngle[Q_NUM_QUADS][Q_NUM_QUADS] =
|
|
{
|
|
0,//Q_BR,Q_BR,
|
|
45,//Q_BR,Q_R,
|
|
90,//Q_BR,Q_TR,
|
|
135,//Q_BR,Q_T,
|
|
180,//Q_BR,Q_TL,
|
|
215,//Q_BR,Q_L,
|
|
270,//Q_BR,Q_BL,
|
|
45,//Q_BR,Q_B,
|
|
45,//Q_R,Q_BR,
|
|
0,//Q_R,Q_R,
|
|
45,//Q_R,Q_TR,
|
|
90,//Q_R,Q_T,
|
|
135,//Q_R,Q_TL,
|
|
180,//Q_R,Q_L,
|
|
215,//Q_R,Q_BL,
|
|
90,//Q_R,Q_B,
|
|
90,//Q_TR,Q_BR,
|
|
45,//Q_TR,Q_R,
|
|
0,//Q_TR,Q_TR,
|
|
45,//Q_TR,Q_T,
|
|
90,//Q_TR,Q_TL,
|
|
135,//Q_TR,Q_L,
|
|
180,//Q_TR,Q_BL,
|
|
135,//Q_TR,Q_B,
|
|
135,//Q_T,Q_BR,
|
|
90,//Q_T,Q_R,
|
|
45,//Q_T,Q_TR,
|
|
0,//Q_T,Q_T,
|
|
45,//Q_T,Q_TL,
|
|
90,//Q_T,Q_L,
|
|
135,//Q_T,Q_BL,
|
|
180,//Q_T,Q_B,
|
|
180,//Q_TL,Q_BR,
|
|
135,//Q_TL,Q_R,
|
|
90,//Q_TL,Q_TR,
|
|
45,//Q_TL,Q_T,
|
|
0,//Q_TL,Q_TL,
|
|
45,//Q_TL,Q_L,
|
|
90,//Q_TL,Q_BL,
|
|
135,//Q_TL,Q_B,
|
|
215,//Q_L,Q_BR,
|
|
180,//Q_L,Q_R,
|
|
135,//Q_L,Q_TR,
|
|
90,//Q_L,Q_T,
|
|
45,//Q_L,Q_TL,
|
|
0,//Q_L,Q_L,
|
|
45,//Q_L,Q_BL,
|
|
90,//Q_L,Q_B,
|
|
270,//Q_BL,Q_BR,
|
|
215,//Q_BL,Q_R,
|
|
180,//Q_BL,Q_TR,
|
|
135,//Q_BL,Q_T,
|
|
90,//Q_BL,Q_TL,
|
|
45,//Q_BL,Q_L,
|
|
0,//Q_BL,Q_BL,
|
|
45,//Q_BL,Q_B,
|
|
45,//Q_B,Q_BR,
|
|
90,//Q_B,Q_R,
|
|
135,//Q_B,Q_TR,
|
|
180,//Q_B,Q_T,
|
|
135,//Q_B,Q_TL,
|
|
90,//Q_B,Q_L,
|
|
45,//Q_B,Q_BL,
|
|
0//Q_B,Q_B,
|
|
};
|
|
|
|
int PM_SaberAttackChainAngle( int move1, int move2 )
|
|
{
|
|
if ( move1 == -1 || move2 == -1 )
|
|
{
|
|
return -1;
|
|
}
|
|
return saberMoveTransitionAngle[saberMoveData[move1].endQuad][saberMoveData[move2].startQuad];
|
|
}
|
|
|
|
qboolean PM_SaberKataDone( int curmove = LS_NONE, int newmove = LS_NONE )
|
|
{
|
|
if ( pm->ps->forceRageRecoveryTime > level.time )
|
|
{//rage recovery, only 1 swing at a time (tired)
|
|
if ( pm->ps->saberAttackChainCount > 0 )
|
|
{//swung once
|
|
return qtrue;
|
|
}
|
|
else
|
|
{//allow one attack
|
|
return qfalse;
|
|
}
|
|
}
|
|
else if ( (pm->ps->forcePowersActive&(1<<FP_RAGE)) )
|
|
{//infinite chaining when raged
|
|
return qfalse;
|
|
}
|
|
else if ( pm->ps->saber[0].maxChain == -1 )
|
|
{
|
|
return qfalse;
|
|
}
|
|
else if ( pm->ps->saber[0].maxChain != 0 )
|
|
{
|
|
if ( pm->ps->saberAttackChainCount >= pm->ps->saber[0].maxChain )
|
|
{
|
|
return qtrue;
|
|
}
|
|
else
|
|
{
|
|
return qfalse;
|
|
}
|
|
}
|
|
|
|
if ( pm->ps->saberAnimLevel == SS_DESANN || pm->ps->saberAnimLevel == SS_TAVION )
|
|
{//desann and tavion can link up as many attacks as they want
|
|
return qfalse;
|
|
}
|
|
//FIXME: instead of random, apply some sort of logical conditions to whether or
|
|
// not you can chain? Like if you were completely missed, you can't chain as much, or...?
|
|
// And/Or based on FP_SABER_OFFENSE level? So number of attacks you can chain
|
|
// increases with your FP_SABER_OFFENSE skill?
|
|
if ( pm->ps->saberAnimLevel == SS_STAFF )
|
|
{
|
|
//TEMP: for now, let staff attacks infinitely chain
|
|
return qfalse;
|
|
/*
|
|
if ( pm->ps->saberAttackChainCount > Q_irand( 3, 4 ) )
|
|
{
|
|
return qtrue;
|
|
}
|
|
else if ( pm->ps->saberAttackChainCount > 0 )
|
|
{
|
|
int chainAngle = PM_SaberAttackChainAngle( curmove, newmove );
|
|
if ( chainAngle < 135 || chainAngle > 215 )
|
|
{//if trying to chain to a move that doesn't continue the momentum
|
|
if ( pm->ps->saberAttackChainCount > 1 )
|
|
{
|
|
return qtrue;
|
|
}
|
|
}
|
|
else if ( chainAngle == 180 )
|
|
{//continues the momentum perfectly, allow it to chain 66% of the time
|
|
if ( pm->ps->saberAttackChainCount > 2 )
|
|
{
|
|
return qtrue;
|
|
}
|
|
}
|
|
else
|
|
{//would continue the movement somewhat, 50% chance of continuing
|
|
if ( pm->ps->saberAttackChainCount > 3 )
|
|
{
|
|
return qtrue;
|
|
}
|
|
}
|
|
}
|
|
*/
|
|
}
|
|
else if ( pm->ps->saberAnimLevel == SS_DUAL )
|
|
{
|
|
//TEMP: for now, let staff attacks infinitely chain
|
|
return qfalse;
|
|
}
|
|
else if ( pm->ps->saberAnimLevel == FORCE_LEVEL_3 )
|
|
{
|
|
if ( curmove == LS_NONE || newmove == LS_NONE )
|
|
{
|
|
if ( pm->ps->saberAnimLevel >= FORCE_LEVEL_3 && pm->ps->saberAttackChainCount > Q_irand( 0, 1 ) )
|
|
{
|
|
return qtrue;
|
|
}
|
|
}
|
|
else if ( pm->ps->saberAttackChainCount > Q_irand( 2, 3 ) )
|
|
{
|
|
return qtrue;
|
|
}
|
|
else if ( pm->ps->saberAttackChainCount > 0 )
|
|
{
|
|
int chainAngle = PM_SaberAttackChainAngle( curmove, newmove );
|
|
if ( chainAngle < 135 || chainAngle > 215 )
|
|
{//if trying to chain to a move that doesn't continue the momentum
|
|
return qtrue;
|
|
}
|
|
else if ( chainAngle == 180 )
|
|
{//continues the momentum perfectly, allow it to chain 66% of the time
|
|
if ( pm->ps->saberAttackChainCount > 1 )
|
|
{
|
|
return qtrue;
|
|
}
|
|
}
|
|
else
|
|
{//would continue the movement somewhat, 50% chance of continuing
|
|
if ( pm->ps->saberAttackChainCount > 2 )
|
|
{
|
|
return qtrue;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{//FIXME: have chainAngle influence fast and medium chains as well?
|
|
if ( (pm->ps->saberAnimLevel == FORCE_LEVEL_2 || pm->ps->saberAnimLevel == SS_DUAL)
|
|
&& pm->ps->saberAttackChainCount > Q_irand( 2, 5 ) )
|
|
{
|
|
return qtrue;
|
|
}
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean PM_CheckEnemyInBack( float backCheckDist )
|
|
{
|
|
if ( !pm->gent || !pm->gent->client )
|
|
{
|
|
return qfalse;
|
|
}
|
|
if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer())
|
|
&& !g_saberAutoAim->integer && pm->cmd.forwardmove >= 0 )
|
|
{//don't auto-backstab
|
|
return qfalse;
|
|
}
|
|
if ( pm->ps->groundEntityNum == ENTITYNUM_NONE )
|
|
{//only when on ground
|
|
return qfalse;
|
|
}
|
|
trace_t trace;
|
|
vec3_t end, fwd, fwdAngles = {0,pm->ps->viewangles[YAW],0};
|
|
|
|
AngleVectors( fwdAngles, fwd, NULL, NULL );
|
|
VectorMA( pm->ps->origin, -backCheckDist, fwd, end );
|
|
|
|
pm->trace( &trace, pm->ps->origin, vec3_origin, vec3_origin, end, pm->ps->clientNum, CONTENTS_SOLID|CONTENTS_BODY );
|
|
if ( trace.fraction < 1.0f && trace.entityNum < ENTITYNUM_WORLD )
|
|
{
|
|
gentity_t *traceEnt = &g_entities[trace.entityNum];
|
|
if ( traceEnt
|
|
&& traceEnt->health > 0
|
|
&& traceEnt->client
|
|
&& traceEnt->client->playerTeam == pm->gent->client->enemyTeam
|
|
&& traceEnt->client->ps.groundEntityNum != ENTITYNUM_NONE )
|
|
{
|
|
if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) )
|
|
{//player
|
|
if ( pm->gent )
|
|
{//set player enemy to traceEnt so he auto-aims at him
|
|
pm->gent->enemy = traceEnt;
|
|
}
|
|
}
|
|
return qtrue;
|
|
}
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
saberMoveName_t PM_PickBackStab( void )
|
|
{
|
|
if ( !pm->gent || !pm->gent->client )
|
|
{
|
|
return LS_READY;
|
|
}
|
|
if ( pm->ps->dualSabers
|
|
&& pm->ps->saber[1].Active() )
|
|
{
|
|
if ( pm->ps->pm_flags & PMF_DUCKED )
|
|
{
|
|
return LS_A_BACK_CR;
|
|
}
|
|
else
|
|
{
|
|
return LS_A_BACK;
|
|
}
|
|
}
|
|
if ( pm->gent->client->ps.saberAnimLevel == SS_TAVION )
|
|
{
|
|
return LS_A_BACKSTAB;
|
|
}
|
|
else if ( pm->gent->client->ps.saberAnimLevel == SS_DESANN )
|
|
{
|
|
if ( pm->ps->saberMove == LS_READY || !Q_irand( 0, 3 ) )
|
|
{
|
|
return LS_A_BACKSTAB;
|
|
}
|
|
else if ( pm->ps->pm_flags & PMF_DUCKED )
|
|
{
|
|
return LS_A_BACK_CR;
|
|
}
|
|
else
|
|
{
|
|
return LS_A_BACK;
|
|
}
|
|
}
|
|
else if ( pm->ps->saberAnimLevel == FORCE_LEVEL_2
|
|
|| pm->ps->saberAnimLevel == SS_DUAL )
|
|
{//using medium attacks or dual sabers
|
|
if ( pm->ps->pm_flags & PMF_DUCKED )
|
|
{
|
|
return LS_A_BACK_CR;
|
|
}
|
|
else
|
|
{
|
|
return LS_A_BACK;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return LS_A_BACKSTAB;
|
|
}
|
|
}
|
|
|
|
saberMoveName_t PM_CheckStabDown( void )
|
|
{
|
|
if ( !pm->gent || !pm->gent->enemy || !pm->gent->enemy->client )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
if ( (pm->ps->saber[0].saberFlags&SFL_NO_STABDOWN) )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
if ( pm->ps->dualSabers
|
|
&& (pm->ps->saber[1].saberFlags&SFL_NO_STABDOWN) )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
if ( pm->ps->clientNum < MAX_CLIENTS || PM_ControlledByPlayer() ) //PLAYER ONLY
|
|
{//player
|
|
if ( G_TryingKataAttack( pm->gent, &pm->cmd ) )//(pm->cmd.buttons&BUTTON_FORCE_FOCUS) )//Holding focus
|
|
{//want to try a special
|
|
return LS_NONE;
|
|
}
|
|
}
|
|
if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) )
|
|
{//player
|
|
if ( pm->ps->groundEntityNum == ENTITYNUM_NONE )//in air
|
|
{//sorry must be on ground (or have just jumped)
|
|
if ( level.time-pm->ps->lastOnGround <= 50 && (pm->ps->pm_flags&PMF_JUMPING) )
|
|
{//just jumped, it's okay
|
|
}
|
|
else
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
}
|
|
/*
|
|
if ( pm->cmd.upmove > 0 )
|
|
{//trying to jump
|
|
}
|
|
else if ( pm->ps->groundEntityNum == ENTITYNUM_NONE //in air
|
|
&& level.time-pm->ps->lastOnGround <= 250 //just left ground
|
|
&& (pm->ps->pm_flags&PMF_JUMPING) )//jumped
|
|
{//just jumped
|
|
}
|
|
else
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
*/
|
|
pm->ps->velocity[2] = 0;
|
|
pm->cmd.upmove = 0;
|
|
}
|
|
else if ( (pm->ps->clientNum >= MAX_CLIENTS&&!PM_ControlledByPlayer()) )
|
|
{//NPC
|
|
if ( pm->ps->groundEntityNum == ENTITYNUM_NONE )//in air
|
|
{//sorry must be on ground (or have just jumped)
|
|
if ( level.time-pm->ps->lastOnGround <= 250 && (pm->ps->pm_flags&PMF_JUMPING) )
|
|
{//just jumped, it's okay
|
|
}
|
|
else
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
}
|
|
if ( !pm->gent->NPC )
|
|
{//wtf???
|
|
return LS_NONE;
|
|
}
|
|
if ( Q_irand( 0, RANK_CAPTAIN ) > pm->gent->NPC->rank )
|
|
{//lower ranks do this less often
|
|
return LS_NONE;
|
|
}
|
|
}
|
|
vec3_t enemyDir, faceFwd, facingAngles = {0, pm->ps->viewangles[YAW], 0};
|
|
AngleVectors( facingAngles, faceFwd, NULL, NULL );
|
|
VectorSubtract( pm->gent->enemy->currentOrigin, pm->ps->origin, enemyDir );
|
|
float enemyZDiff = enemyDir[2];
|
|
enemyDir[2] = 0;
|
|
float enemyHDist = VectorNormalize( enemyDir )-(pm->gent->maxs[0]+pm->gent->enemy->maxs[0]);
|
|
float dot = DotProduct( enemyDir, faceFwd );
|
|
|
|
if ( //(pm->ps->clientNum < MAX_CLIENTS || PM_ControlledByPlayer())
|
|
dot > 0.65f
|
|
//&& enemyHDist >= 32 //was 48
|
|
&& enemyHDist <= 164//was 112
|
|
&& PM_InKnockDownOnGround( &pm->gent->enemy->client->ps )//still on ground
|
|
&& !PM_InGetUpNoRoll( &pm->gent->enemy->client->ps )//not getting up yet
|
|
&& enemyZDiff <= 20 )
|
|
{//guy is on the ground below me, do a top-down attack
|
|
if ( pm->gent->enemy->s.number >= MAX_CLIENTS
|
|
|| !G_ControlledByPlayer( pm->gent->enemy ) )
|
|
{//don't get up while I'm doing this
|
|
//stop them from trying to get up for at least another 3 seconds
|
|
TIMER_Set( pm->gent->enemy, "noGetUpStraight", 3000 );
|
|
}
|
|
//pick the right anim
|
|
if ( pm->ps->saberAnimLevel == SS_DUAL
|
|
|| (pm->ps->dualSabers&&pm->ps->saber[1].Active()) )
|
|
{
|
|
return LS_STABDOWN_DUAL;
|
|
}
|
|
else if ( pm->ps->saberAnimLevel == SS_STAFF )
|
|
{
|
|
return LS_STABDOWN_STAFF;
|
|
}
|
|
else
|
|
{
|
|
return LS_STABDOWN;
|
|
}
|
|
}
|
|
return LS_NONE;
|
|
}
|
|
|
|
extern saberMoveName_t PM_NPCSaberAttackFromQuad( int quad );
|
|
saberMoveName_t PM_SaberFlipOverAttackMove( void );
|
|
saberMoveName_t PM_AttackForEnemyPos( qboolean allowFB, qboolean allowStabDown )
|
|
{
|
|
saberMoveName_t autoMove = LS_INVALID;
|
|
|
|
if( !pm->gent->enemy )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
|
|
vec3_t enemy_org, enemyDir, faceFwd, faceRight, faceUp, facingAngles = {0, pm->ps->viewangles[YAW], 0};
|
|
AngleVectors( facingAngles, faceFwd, faceRight, faceUp );
|
|
//FIXME: predict enemy position?
|
|
if ( pm->gent->enemy->client )
|
|
{
|
|
//VectorCopy( pm->gent->enemy->currentOrigin, enemy_org );
|
|
//HMM... using this will adjust for bbox size, so let's do that...
|
|
vec3_t size;
|
|
VectorSubtract( pm->gent->enemy->absmax, pm->gent->enemy->absmin, size );
|
|
VectorMA( pm->gent->enemy->absmin, 0.5, size, enemy_org );
|
|
|
|
VectorSubtract( pm->gent->enemy->client->renderInfo.eyePoint, pm->ps->origin, enemyDir );
|
|
}
|
|
else
|
|
{
|
|
if ( pm->gent->enemy->bmodel && VectorCompare( vec3_origin, pm->gent->enemy->currentOrigin ) )
|
|
{//a brush model without an origin brush
|
|
vec3_t size;
|
|
VectorSubtract( pm->gent->enemy->absmax, pm->gent->enemy->absmin, size );
|
|
VectorMA( pm->gent->enemy->absmin, 0.5, size, enemy_org );
|
|
}
|
|
else
|
|
{
|
|
VectorCopy( pm->gent->enemy->currentOrigin, enemy_org );
|
|
}
|
|
VectorSubtract( enemy_org, pm->ps->origin, enemyDir );
|
|
}
|
|
float enemyZDiff = enemyDir[2];
|
|
float enemyDist = VectorNormalize( enemyDir );
|
|
float dot = DotProduct( enemyDir, faceFwd );
|
|
if ( dot > 0 )
|
|
{//enemy is in front
|
|
if ( allowStabDown )
|
|
{//okay to try this
|
|
saberMoveName_t stabDownMove = PM_CheckStabDown();
|
|
if ( stabDownMove != LS_NONE )
|
|
{
|
|
return stabDownMove;
|
|
}
|
|
}
|
|
if ( (pm->ps->clientNum < MAX_CLIENTS || PM_ControlledByPlayer())
|
|
&& dot > 0.65f
|
|
&& enemyDist <= 64 && pm->gent->enemy->client
|
|
&& (enemyZDiff <= 20 || PM_InKnockDownOnGround( &pm->gent->enemy->client->ps ) || PM_CrouchAnim( pm->gent->enemy->client->ps.legsAnim ) ) )
|
|
{//swing down at them
|
|
return LS_A_T2B;
|
|
}
|
|
if ( allowFB )
|
|
{//directly in front anim allowed
|
|
if ( !(pm->ps->saber[0].saberFlags&SFL_NO_BACK_ATTACK)
|
|
&& (!pm->ps->dualSabers || !(pm->ps->saber[1].saberFlags&SFL_NO_BACK_ATTACK)) )
|
|
{//okay to do backstabs with this saber
|
|
if ( enemyDist > 200 || pm->gent->enemy->health <= 0 )
|
|
{//hmm, look in back for an enemy
|
|
if ( pm->ps->clientNum && !PM_ControlledByPlayer() )
|
|
{//player should never do this automatically
|
|
if ( pm->ps->groundEntityNum != ENTITYNUM_NONE )
|
|
{//only when on ground
|
|
if ( pm->gent && pm->gent->client && pm->gent->NPC && pm->gent->NPC->rank >= RANK_LT_JG && Q_irand( 0, pm->gent->NPC->rank ) > RANK_ENSIGN )
|
|
{//only fencers and higher can do this, higher rank does it more
|
|
if ( PM_CheckEnemyInBack( 100 ) )
|
|
{
|
|
return PM_PickBackStab();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
//this is the default only if they're *right* in front...
|
|
if ( (pm->ps->clientNum&&!PM_ControlledByPlayer())
|
|
|| ((pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && cg.renderingThirdPerson && !cg.zoomMode) )
|
|
{//NPC or player not in 1st person
|
|
if ( PM_CheckFlipOverAttackMove( qtrue ) )
|
|
{//enemy must be close and in front
|
|
return PM_SaberFlipOverAttackMove();
|
|
}
|
|
}
|
|
if ( PM_CheckLungeAttackMove() )
|
|
{//NPC
|
|
autoMove = PM_SaberLungeAttackMove( qtrue );
|
|
}
|
|
else
|
|
{
|
|
autoMove = LS_A_T2B;
|
|
}
|
|
}
|
|
else
|
|
{//pick a random one
|
|
if ( Q_irand( 0, 1 ) )
|
|
{
|
|
autoMove = LS_A_TR2BL;
|
|
}
|
|
else
|
|
{
|
|
autoMove = LS_A_TL2BR;
|
|
}
|
|
}
|
|
float dotR = DotProduct( enemyDir, faceRight );
|
|
if ( dotR > 0.35 )
|
|
{//enemy is to far right
|
|
autoMove = LS_A_L2R;
|
|
}
|
|
else if ( dotR < -0.35 )
|
|
{//far left
|
|
autoMove = LS_A_R2L;
|
|
}
|
|
else if ( dotR > 0.15 )
|
|
{//enemy is to near right
|
|
autoMove = LS_A_TR2BL;
|
|
}
|
|
else if ( dotR < -0.15 )
|
|
{//near left
|
|
autoMove = LS_A_TL2BR;
|
|
}
|
|
if ( DotProduct( enemyDir, faceUp ) > 0.5 )
|
|
{//enemy is above me
|
|
if ( autoMove == LS_A_TR2BL )
|
|
{
|
|
autoMove = LS_A_BL2TR;
|
|
}
|
|
else if ( autoMove == LS_A_TL2BR )
|
|
{
|
|
autoMove = LS_A_BR2TL;
|
|
}
|
|
}
|
|
}
|
|
else if ( allowFB )
|
|
{//back attack allowed
|
|
//if ( !PM_InKnockDown( pm->ps ) )
|
|
if ( !(pm->ps->saber[0].saberFlags&SFL_NO_BACK_ATTACK)
|
|
&& (!pm->ps->dualSabers || !(pm->ps->saber[1].saberFlags&SFL_NO_BACK_ATTACK)) )
|
|
{//okay to do backstabs with this saber
|
|
if ( pm->ps->groundEntityNum != ENTITYNUM_NONE )
|
|
{//only when on ground
|
|
if ( !pm->gent->enemy->client || pm->gent->enemy->client->ps.groundEntityNum != ENTITYNUM_NONE )
|
|
{//enemy not a client or is a client and on ground
|
|
if ( dot < -0.75f
|
|
&& enemyDist < 128
|
|
&& (pm->ps->saberAnimLevel == SS_FAST || pm->ps->saberAnimLevel == SS_STAFF || (pm->gent->client &&(pm->gent->client->NPC_class == CLASS_TAVION||pm->gent->client->NPC_class == CLASS_ALORA)&&Q_irand(0,2))) )
|
|
{//fast back-stab
|
|
if ( !(pm->ps->pm_flags&PMF_DUCKED) && pm->cmd.upmove >= 0 )
|
|
{//can't do it while ducked?
|
|
if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) || (pm->gent->NPC && pm->gent->NPC->rank >= RANK_LT_JG) )
|
|
{//only fencers and above can do this
|
|
autoMove = LS_A_BACKSTAB;
|
|
}
|
|
}
|
|
}
|
|
else if ( pm->ps->saberAnimLevel != SS_FAST
|
|
&& pm->ps->saberAnimLevel != SS_STAFF )
|
|
{//higher level back spin-attacks
|
|
if ( (pm->ps->clientNum&&!PM_ControlledByPlayer()) || ((pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && cg.renderingThirdPerson && !cg.zoomMode) )
|
|
{
|
|
if ( (pm->ps->pm_flags&PMF_DUCKED) || pm->cmd.upmove < 0 )
|
|
{
|
|
autoMove = LS_A_BACK_CR;
|
|
}
|
|
else
|
|
{
|
|
autoMove = LS_A_BACK;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return autoMove;
|
|
}
|
|
|
|
qboolean PM_InSecondaryStyle( void )
|
|
{
|
|
if ( pm->ps->saber[0].numBlades > 1
|
|
&& pm->ps->saber[0].singleBladeStyle
|
|
&& (pm->ps->saber[0].stylesForbidden&(1<<pm->ps->saber[0].singleBladeStyle))
|
|
&& pm->ps->saberAnimLevel == pm->ps->saber[0].singleBladeStyle )
|
|
{
|
|
return qtrue;
|
|
}
|
|
|
|
if ( pm->ps->dualSabers
|
|
&& !pm->ps->saber[1].Active() )//pm->ps->saberAnimLevel != SS_DUAL )
|
|
{
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
saberMoveName_t PM_SaberLungeAttackMove( qboolean fallbackToNormalLunge )
|
|
{
|
|
G_DrainPowerForSpecialMove( pm->gent, FP_SABER_OFFENSE, SABER_ALT_ATTACK_POWER_FB );
|
|
|
|
//see if we have an overridden (or cancelled) lunge move
|
|
if ( pm->ps->saber[0].lungeAtkMove != LS_INVALID )
|
|
{
|
|
if ( pm->ps->saber[0].lungeAtkMove != LS_NONE )
|
|
{
|
|
return (saberMoveName_t)pm->ps->saber[0].lungeAtkMove;
|
|
}
|
|
}
|
|
if ( pm->ps->dualSabers )
|
|
{
|
|
if ( pm->ps->saber[1].lungeAtkMove != LS_INVALID )
|
|
{
|
|
if ( pm->ps->saber[1].lungeAtkMove != LS_NONE )
|
|
{
|
|
return (saberMoveName_t)pm->ps->saber[1].lungeAtkMove;
|
|
}
|
|
}
|
|
}
|
|
//no overrides, cancelled?
|
|
if ( pm->ps->saber[0].lungeAtkMove == LS_NONE )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
if ( pm->ps->dualSabers )
|
|
{
|
|
if ( pm->ps->saber[1].lungeAtkMove == LS_NONE )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
}
|
|
//do normal checks
|
|
if ( pm->gent->client->NPC_class == CLASS_ALORA && !Q_irand( 0, 3 ) )
|
|
{//alora NPC
|
|
return LS_SPINATTACK_ALORA;
|
|
}
|
|
else
|
|
{
|
|
if ( pm->ps->dualSabers )
|
|
{
|
|
return LS_SPINATTACK_DUAL;
|
|
}
|
|
switch ( pm->ps->saberAnimLevel )
|
|
{
|
|
case SS_DUAL:
|
|
return LS_SPINATTACK_DUAL;
|
|
break;
|
|
case SS_STAFF:
|
|
return LS_SPINATTACK;
|
|
break;
|
|
default://normal lunge
|
|
if ( fallbackToNormalLunge )
|
|
{
|
|
vec3_t fwdAngles, jumpFwd;
|
|
|
|
VectorCopy( pm->ps->viewangles, fwdAngles );
|
|
fwdAngles[PITCH] = fwdAngles[ROLL] = 0;
|
|
//do the lunge
|
|
AngleVectors( fwdAngles, jumpFwd, NULL, NULL );
|
|
VectorScale( jumpFwd, 150, pm->ps->velocity );
|
|
pm->ps->velocity[2] = 50;
|
|
PM_AddEvent( EV_JUMP );
|
|
|
|
return LS_A_LUNGE;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
return LS_NONE;
|
|
}
|
|
|
|
qboolean PM_CheckLungeAttackMove( void )
|
|
{
|
|
//check to see if it's cancelled?
|
|
if ( pm->ps->saber[0].lungeAtkMove == LS_NONE )
|
|
{
|
|
if ( pm->ps->dualSabers )
|
|
{
|
|
if ( pm->ps->saber[1].lungeAtkMove == LS_NONE
|
|
|| pm->ps->saber[1].lungeAtkMove == LS_INVALID )
|
|
{
|
|
return qfalse;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return qfalse;
|
|
}
|
|
}
|
|
if ( pm->ps->dualSabers )
|
|
{
|
|
if ( pm->ps->saber[1].lungeAtkMove == LS_NONE )
|
|
{
|
|
if ( pm->ps->saber[0].lungeAtkMove == LS_NONE
|
|
|| pm->ps->saber[0].lungeAtkMove == LS_INVALID )
|
|
{
|
|
return qfalse;
|
|
}
|
|
}
|
|
}
|
|
//do normal checks
|
|
if ( pm->ps->saberAnimLevel == SS_FAST//fast
|
|
|| pm->ps->saberAnimLevel == SS_DUAL//dual
|
|
|| pm->ps->saberAnimLevel == SS_STAFF //staff
|
|
|| pm->ps->saberAnimLevel == SS_DESANN
|
|
|| pm->ps->dualSabers )
|
|
{//alt+back+attack using fast, dual or staff attacks
|
|
if ( pm->ps->clientNum >= MAX_CLIENTS && !PM_ControlledByPlayer() )
|
|
{//NPC
|
|
if ( pm->cmd.upmove < 0 || (pm->ps->pm_flags&PMF_DUCKED) )
|
|
{//ducking
|
|
if ( pm->ps->legsAnim == BOTH_STAND2
|
|
|| pm->ps->legsAnim == BOTH_SABERFAST_STANCE
|
|
|| pm->ps->legsAnim == BOTH_SABERSLOW_STANCE
|
|
|| pm->ps->legsAnim == BOTH_SABERSTAFF_STANCE
|
|
|| pm->ps->legsAnim == BOTH_SABERDUAL_STANCE
|
|
|| (level.time-pm->ps->lastStationary) <= 500 )
|
|
{//standing or just stopped standing
|
|
if ( pm->gent
|
|
&& pm->gent->NPC //NPC
|
|
&& pm->gent->NPC->rank >= RANK_LT_JG //high rank
|
|
&& ( pm->gent->NPC->rank == RANK_LT_JG || Q_irand( -3, pm->gent->NPC->rank ) >= RANK_LT_JG )//Q_irand( 0, pm->gent->NPC->rank ) >= RANK_LT_JG )
|
|
&& !Q_irand( 0, 3-g_spskill->integer ) )
|
|
{//only fencer and higher can do this
|
|
if ( pm->ps->saberAnimLevel == SS_DESANN )
|
|
{
|
|
if ( !Q_irand( 0, 4 ) )
|
|
{
|
|
return qtrue;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return qtrue;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{//player
|
|
if ( G_TryingLungeAttack( pm->gent, &pm->cmd )
|
|
&& G_EnoughPowerForSpecialMove( pm->ps->forcePower, SABER_ALT_ATTACK_POWER_FB )/*pm->ps->forcePower >= SABER_ALT_ATTACK_POWER_FB*/ )//have enough force power to pull it off
|
|
{
|
|
return qtrue;
|
|
}
|
|
}
|
|
}
|
|
|
|
return qfalse;
|
|
}
|
|
|
|
saberMoveName_t PM_SaberJumpForwardAttackMove( void )
|
|
{
|
|
G_DrainPowerForSpecialMove( pm->gent, FP_LEVITATION, SABER_ALT_ATTACK_POWER_FB );
|
|
|
|
//see if we have an overridden (or cancelled) kata move
|
|
if ( pm->ps->saber[0].jumpAtkFwdMove != LS_INVALID )
|
|
{
|
|
if ( pm->ps->saber[0].jumpAtkFwdMove != LS_NONE )
|
|
{
|
|
return (saberMoveName_t)pm->ps->saber[0].jumpAtkFwdMove;
|
|
}
|
|
}
|
|
if ( pm->ps->dualSabers )
|
|
{
|
|
if ( pm->ps->saber[1].jumpAtkFwdMove != LS_INVALID )
|
|
{
|
|
if ( pm->ps->saber[1].jumpAtkFwdMove != LS_NONE )
|
|
{
|
|
return (saberMoveName_t)pm->ps->saber[1].jumpAtkFwdMove;
|
|
}
|
|
}
|
|
}
|
|
//no overrides, cancelled?
|
|
if ( pm->ps->saber[0].jumpAtkFwdMove == LS_NONE )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
if ( pm->ps->dualSabers )
|
|
{
|
|
if ( pm->ps->saber[1].jumpAtkFwdMove == LS_NONE )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
}
|
|
if ( pm->ps->saberAnimLevel == SS_DUAL
|
|
|| pm->ps->saberAnimLevel == SS_STAFF )
|
|
{
|
|
pm->cmd.upmove = 0;//no jump just yet
|
|
|
|
if ( pm->ps->saberAnimLevel == SS_STAFF )
|
|
{
|
|
if ( Q_irand(0, 1) )
|
|
{
|
|
return LS_JUMPATTACK_STAFF_LEFT;
|
|
}
|
|
else
|
|
{
|
|
return LS_JUMPATTACK_STAFF_RIGHT;
|
|
}
|
|
}
|
|
|
|
return LS_JUMPATTACK_DUAL;
|
|
}
|
|
else
|
|
{
|
|
vec3_t fwdAngles, jumpFwd;
|
|
|
|
VectorCopy( pm->ps->viewangles, fwdAngles );
|
|
fwdAngles[PITCH] = fwdAngles[ROLL] = 0;
|
|
AngleVectors( fwdAngles, jumpFwd, NULL, NULL );
|
|
VectorScale( jumpFwd, 200, pm->ps->velocity );
|
|
pm->ps->velocity[2] = 180;
|
|
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;
|
|
|
|
//FIXME: NPCs yell?
|
|
PM_AddEvent( EV_JUMP );
|
|
G_SoundOnEnt( pm->gent, CHAN_BODY, "sound/weapons/force/jump.wav" );
|
|
pm->cmd.upmove = 0;
|
|
|
|
return LS_A_JUMP_T__B_;
|
|
}
|
|
}
|
|
|
|
qboolean PM_CheckJumpForwardAttackMove( void )
|
|
{
|
|
if ( pm->ps->clientNum < MAX_CLIENTS
|
|
&& PM_InSecondaryStyle() )
|
|
{
|
|
return qfalse;
|
|
}
|
|
|
|
//check to see if it's cancelled?
|
|
if ( pm->ps->saber[0].jumpAtkFwdMove == LS_NONE )
|
|
{
|
|
if ( pm->ps->dualSabers )
|
|
{
|
|
if ( pm->ps->saber[1].jumpAtkFwdMove == LS_NONE
|
|
|| pm->ps->saber[1].jumpAtkFwdMove == LS_INVALID )
|
|
{
|
|
return qfalse;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return qfalse;
|
|
}
|
|
}
|
|
if ( pm->ps->dualSabers )
|
|
{
|
|
if ( pm->ps->saber[1].jumpAtkFwdMove == LS_NONE )
|
|
{
|
|
if ( pm->ps->saber[0].jumpAtkFwdMove == LS_NONE
|
|
|| pm->ps->saber[0].jumpAtkFwdMove == LS_INVALID )
|
|
{
|
|
return qfalse;
|
|
}
|
|
}
|
|
}
|
|
//do normal checks
|
|
|
|
if ( pm->cmd.forwardmove > 0 //going forward
|
|
&& pm->ps->forceRageRecoveryTime < pm->cmd.serverTime //not in a force Rage recovery period
|
|
&& pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_1 //can force jump
|
|
&& pm->gent && !(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
|
|
&& (pm->ps->groundEntityNum != ENTITYNUM_NONE||level.time-pm->ps->lastOnGround<=250) //on ground or just jumped (if not player)
|
|
)
|
|
{
|
|
if ( pm->ps->saberAnimLevel == SS_DUAL
|
|
|| pm->ps->saberAnimLevel == SS_STAFF )
|
|
{//dual and staff
|
|
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
|
|
{
|
|
if ( pm->ps->clientNum >= MAX_CLIENTS && !PM_ControlledByPlayer() )
|
|
{//NPC
|
|
if ( pm->cmd.upmove > 0 || (pm->ps->pm_flags&PMF_JUMPING) )//jumping NPC
|
|
{
|
|
if ( pm->gent
|
|
&& pm->gent->NPC
|
|
&& (pm->gent->NPC->rank==RANK_CREWMAN||pm->gent->NPC->rank>=RANK_LT) )
|
|
{
|
|
return qtrue;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{//PLAYER
|
|
if ( G_TryingJumpForwardAttack( pm->gent, &pm->cmd )
|
|
&& G_EnoughPowerForSpecialMove( pm->ps->forcePower, SABER_ALT_ATTACK_POWER_FB ) )//have enough power to attack
|
|
{
|
|
return qtrue;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
//check strong
|
|
else if ( pm->ps->saberAnimLevel == SS_STRONG //strong style
|
|
|| pm->ps->saberAnimLevel == SS_DESANN )//desann
|
|
{
|
|
if ( //&& !PM_InKnockDown( pm->ps )
|
|
!pm->ps->dualSabers
|
|
//&& (pm->ps->legsAnim == BOTH_STAND2||pm->ps->legsAnim == BOTH_SABERFAST_STANCE||pm->ps->legsAnim == BOTH_SABERSLOW_STANCE||level.time-pm->ps->lastStationary<=500)//standing or just started moving
|
|
)
|
|
{//strong attack: jump-hack
|
|
/*
|
|
if ( pm->ps->legsAnim == BOTH_STAND2
|
|
|| pm->ps->legsAnim == BOTH_SABERFAST_STANCE
|
|
|| pm->ps->legsAnim == BOTH_SABERSLOW_STANCE
|
|
|| level.time-pm->ps->lastStationary <= 250 )//standing or just started moving
|
|
*/
|
|
if ( pm->ps->clientNum >= MAX_CLIENTS && !PM_ControlledByPlayer() )
|
|
{//NPC
|
|
if ( pm->cmd.upmove > 0 || (pm->ps->pm_flags&PMF_JUMPING) )//NPC jumping
|
|
{
|
|
if ( pm->gent
|
|
&& pm->gent->NPC
|
|
&& (pm->gent->NPC->rank==RANK_CREWMAN||pm->gent->NPC->rank>=RANK_LT) )
|
|
{//only acrobat or boss and higher can do this
|
|
if ( pm->ps->legsAnim == BOTH_STAND2
|
|
|| pm->ps->legsAnim == BOTH_SABERFAST_STANCE
|
|
|| pm->ps->legsAnim == BOTH_SABERSLOW_STANCE
|
|
|| level.time-pm->ps->lastStationary <= 250 )
|
|
{//standing or just started moving
|
|
if ( pm->gent->client
|
|
&& pm->gent->client->NPC_class == CLASS_DESANN )
|
|
{
|
|
if ( !Q_irand( 0, 1 ) )
|
|
{
|
|
return qtrue;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return qtrue;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{//player
|
|
if ( G_TryingJumpForwardAttack( pm->gent, &pm->cmd )
|
|
&& G_EnoughPowerForSpecialMove( pm->ps->forcePower, SABER_ALT_ATTACK_POWER_FB ) )
|
|
{
|
|
return qtrue;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
saberMoveName_t PM_SaberFlipOverAttackMove( void )
|
|
{
|
|
//see if we have an overridden (or cancelled) kata move
|
|
if ( pm->ps->saber[0].jumpAtkFwdMove != LS_INVALID )
|
|
{
|
|
if ( pm->ps->saber[0].jumpAtkFwdMove != LS_NONE )
|
|
{
|
|
return (saberMoveName_t)pm->ps->saber[0].jumpAtkFwdMove;
|
|
}
|
|
}
|
|
if ( pm->ps->dualSabers )
|
|
{
|
|
if ( pm->ps->saber[1].jumpAtkFwdMove != LS_INVALID )
|
|
{
|
|
if ( pm->ps->saber[1].jumpAtkFwdMove != LS_NONE )
|
|
{
|
|
return (saberMoveName_t)pm->ps->saber[1].jumpAtkFwdMove;
|
|
}
|
|
}
|
|
}
|
|
//no overrides, cancelled?
|
|
if ( pm->ps->saber[0].jumpAtkFwdMove == LS_NONE )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
if ( pm->ps->dualSabers )
|
|
{
|
|
if ( pm->ps->saber[1].jumpAtkFwdMove == LS_NONE )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
}
|
|
//FIXME: check above for room enough to jump!
|
|
//FIXME: while in this jump, keep velocity[2] at a minimum until the end of the anim
|
|
vec3_t fwdAngles, jumpFwd;
|
|
|
|
VectorCopy( pm->ps->viewangles, fwdAngles );
|
|
fwdAngles[PITCH] = fwdAngles[ROLL] = 0;
|
|
AngleVectors( fwdAngles, jumpFwd, NULL, NULL );
|
|
VectorScale( jumpFwd, 150, pm->ps->velocity );
|
|
pm->ps->velocity[2] = 250;
|
|
//250 is normalized for a standing enemy at your z level, about 64 tall... adjust for actual maxs[2]-mins[2] of enemy and for zdiff in origins
|
|
if ( pm->gent && pm->gent->enemy )
|
|
{ //go higher for taller enemies
|
|
pm->ps->velocity[2] *= (pm->gent->enemy->maxs[2]-pm->gent->enemy->mins[2])/64.0f;
|
|
//go higher for enemies higher than you, lower for those lower than you
|
|
float zDiff = pm->gent->enemy->currentOrigin[2] - pm->ps->origin[2];
|
|
pm->ps->velocity[2] += (zDiff)*1.5f;
|
|
//clamp to decent-looking values
|
|
//FIXME: still jump too low sometimes
|
|
if ( zDiff <= 0 && pm->ps->velocity[2] < 200 )
|
|
{//if we're on same level, don't let me jump so low, I clip into the ground
|
|
pm->ps->velocity[2] = 200;
|
|
}
|
|
else if ( pm->ps->velocity[2] < 50 )
|
|
{
|
|
pm->ps->velocity[2] = 50;
|
|
}
|
|
else if ( pm->ps->velocity[2] > 400 )
|
|
{
|
|
pm->ps->velocity[2] = 400;
|
|
}
|
|
}
|
|
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;
|
|
|
|
//FIXME: NPCs yell?
|
|
PM_AddEvent( EV_JUMP );
|
|
G_SoundOnEnt( pm->gent, CHAN_BODY, "sound/weapons/force/jump.wav" );
|
|
pm->cmd.upmove = 0;
|
|
//FIXME: don't allow this to land on other people
|
|
|
|
pm->gent->angle = pm->ps->viewangles[YAW];//so we know what yaw we started this at
|
|
|
|
G_DrainPowerForSpecialMove( pm->gent, FP_LEVITATION, SABER_ALT_ATTACK_POWER_FB );
|
|
|
|
if ( Q_irand( 0, 1 ) )
|
|
{
|
|
return LS_A_FLIP_STAB;
|
|
}
|
|
else
|
|
{
|
|
return LS_A_FLIP_SLASH;
|
|
}
|
|
}
|
|
|
|
qboolean PM_CheckFlipOverAttackMove( qboolean checkEnemy )
|
|
{
|
|
if ( pm->ps->clientNum < MAX_CLIENTS
|
|
&& PM_InSecondaryStyle() )
|
|
{
|
|
return qfalse;
|
|
}
|
|
//check to see if it's cancelled?
|
|
if ( pm->ps->saber[0].jumpAtkFwdMove == LS_NONE )
|
|
{
|
|
if ( pm->ps->dualSabers )
|
|
{
|
|
if ( pm->ps->saber[1].jumpAtkFwdMove == LS_NONE
|
|
|| pm->ps->saber[1].jumpAtkFwdMove == LS_INVALID )
|
|
{
|
|
return qfalse;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return qfalse;
|
|
}
|
|
}
|
|
if ( pm->ps->dualSabers )
|
|
{
|
|
if ( pm->ps->saber[1].jumpAtkFwdMove == LS_NONE )
|
|
{
|
|
if ( pm->ps->saber[0].jumpAtkFwdMove == LS_NONE
|
|
|| pm->ps->saber[0].jumpAtkFwdMove == LS_INVALID )
|
|
{
|
|
return qfalse;
|
|
}
|
|
}
|
|
}
|
|
//do normal checks
|
|
|
|
if ( (pm->ps->saberAnimLevel == SS_MEDIUM //medium
|
|
|| pm->ps->saberAnimLevel == SS_TAVION )//tavion
|
|
&& pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_1 //can force jump
|
|
&& !(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
|
|
&& (pm->ps->groundEntityNum != ENTITYNUM_NONE||level.time-pm->ps->lastOnGround<=250) //on ground or just jumped
|
|
)
|
|
{
|
|
qboolean tryMove = qfalse;
|
|
if ( pm->ps->clientNum >= MAX_CLIENTS && !PM_ControlledByPlayer() )
|
|
{//NPC
|
|
if ( pm->cmd.upmove > 0//want to jump
|
|
|| (pm->ps->pm_flags&PMF_JUMPING) )//jumping
|
|
{//flip over-forward down-attack
|
|
if ( (pm->gent->NPC
|
|
&& (pm->gent->NPC->rank==RANK_CREWMAN||pm->gent->NPC->rank>=RANK_LT)
|
|
&& !Q_irand(0, 2) ) )//NPC who can do this, 33% chance
|
|
{//only player or acrobat or boss and higher can do this
|
|
tryMove = qtrue;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{//player
|
|
if ( G_TryingJumpForwardAttack( pm->gent, &pm->cmd )
|
|
&& G_EnoughPowerForSpecialMove( pm->ps->forcePower, SABER_ALT_ATTACK_POWER_FB ) )//have enough power
|
|
{
|
|
if ( !pm->cmd.rightmove )
|
|
{
|
|
if ( pm->ps->legsAnim == BOTH_JUMP1
|
|
|| pm->ps->legsAnim == BOTH_FORCEJUMP1
|
|
|| pm->ps->legsAnim == BOTH_INAIR1
|
|
|| pm->ps->legsAnim == BOTH_FORCEINAIR1 )
|
|
{//in a non-flip forward jump
|
|
tryMove = qtrue;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( tryMove )
|
|
{
|
|
if ( !checkEnemy )
|
|
{//based just on command input
|
|
return qtrue;
|
|
}
|
|
else
|
|
{//based on presence of enemy
|
|
if ( pm->gent->enemy )//have an enemy
|
|
{
|
|
vec3_t fwdAngles = {0,pm->ps->viewangles[YAW],0};
|
|
if ( pm->gent->enemy->health > 0
|
|
&& pm->ps->forceRageRecoveryTime < pm->cmd.serverTime //not in a force Rage recovery period
|
|
&& pm->gent->enemy->maxs[2] > 12
|
|
&& (!pm->gent->enemy->client || !PM_InKnockDownOnGround( &pm->gent->enemy->client->ps ) )
|
|
&& DistanceSquared( pm->gent->currentOrigin, pm->gent->enemy->currentOrigin ) < 10000
|
|
&& InFront( pm->gent->enemy->currentOrigin, pm->gent->currentOrigin, fwdAngles, 0.3f ) )
|
|
{//enemy must be alive, not low to ground, close and in front
|
|
return qtrue;
|
|
}
|
|
}
|
|
return qfalse;
|
|
}
|
|
}
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
saberMoveName_t PM_SaberBackflipAttackMove( void )
|
|
{
|
|
//see if we have an overridden (or cancelled) kata move
|
|
if ( pm->ps->saber[0].jumpAtkBackMove != LS_INVALID )
|
|
{
|
|
if ( pm->ps->saber[0].jumpAtkBackMove != LS_NONE )
|
|
{
|
|
return (saberMoveName_t)pm->ps->saber[0].jumpAtkBackMove;
|
|
}
|
|
}
|
|
if ( pm->ps->dualSabers )
|
|
{
|
|
if ( pm->ps->saber[1].jumpAtkBackMove != LS_INVALID )
|
|
{
|
|
if ( pm->ps->saber[1].jumpAtkBackMove != LS_NONE )
|
|
{
|
|
return (saberMoveName_t)pm->ps->saber[1].jumpAtkBackMove;
|
|
}
|
|
}
|
|
}
|
|
//no overrides, cancelled?
|
|
if ( pm->ps->saber[0].jumpAtkBackMove == LS_NONE )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
if ( pm->ps->dualSabers )
|
|
{
|
|
if ( pm->ps->saber[1].jumpAtkBackMove == LS_NONE )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
}
|
|
pm->cmd.upmove = 0;//no jump just yet
|
|
return LS_A_BACKFLIP_ATK;
|
|
}
|
|
|
|
qboolean PM_CheckBackflipAttackMove( void )
|
|
{
|
|
if ( pm->ps->clientNum < MAX_CLIENTS
|
|
&& PM_InSecondaryStyle() )
|
|
{
|
|
return qfalse;
|
|
}
|
|
|
|
//check to see if it's cancelled?
|
|
if ( pm->ps->saber[0].jumpAtkBackMove == LS_NONE )
|
|
{
|
|
if ( pm->ps->dualSabers )
|
|
{
|
|
if ( pm->ps->saber[1].jumpAtkBackMove == LS_NONE
|
|
|| pm->ps->saber[1].jumpAtkBackMove == LS_INVALID )
|
|
{
|
|
return qfalse;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return qfalse;
|
|
}
|
|
}
|
|
if ( pm->ps->dualSabers )
|
|
{
|
|
if ( pm->ps->saber[1].jumpAtkBackMove == LS_NONE )
|
|
{
|
|
if ( pm->ps->saber[0].jumpAtkBackMove == LS_NONE
|
|
|| pm->ps->saber[0].jumpAtkBackMove == LS_INVALID )
|
|
{
|
|
return qfalse;
|
|
}
|
|
}
|
|
}
|
|
//do normal checks
|
|
|
|
if ( pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_1 //can force jump
|
|
&& pm->ps->forceRageRecoveryTime < pm->cmd.serverTime //not in a force Rage recovery period
|
|
&& pm->gent && !(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
|
|
//&& (pm->ps->legsAnim == BOTH_SABERSTAFF_STANCE || level.time-pm->ps->lastStationary<=250)//standing or just started moving
|
|
&& (pm->ps->groundEntityNum != ENTITYNUM_NONE||level.time-pm->ps->lastOnGround<=250) )//on ground or just jumped (if not player)
|
|
{
|
|
if ( pm->cmd.forwardmove < 0 //moving backwards
|
|
&& pm->ps->saberAnimLevel == SS_STAFF //using staff
|
|
&& (pm->cmd.upmove > 0 || (pm->ps->pm_flags&PMF_JUMPING)) )//jumping
|
|
{//jumping backwards and using staff
|
|
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 already attacking
|
|
if ( pm->ps->clientNum >= MAX_CLIENTS && !PM_ControlledByPlayer() )
|
|
{//NPC
|
|
if ( pm->gent
|
|
&& pm->gent->NPC
|
|
&& (pm->gent->NPC->rank==RANK_CREWMAN||pm->gent->NPC->rank>=RANK_LT) )
|
|
{//acrobat or boss and higher can do this
|
|
return qtrue;
|
|
}
|
|
}
|
|
else
|
|
{//player
|
|
return qtrue;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
saberMoveName_t PM_CheckDualSpinProtect( void )
|
|
{
|
|
if ( pm->ps->clientNum < MAX_CLIENTS
|
|
&& PM_InSecondaryStyle() )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
|
|
//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 )
|
|
{
|
|
return (saberMoveName_t)pm->ps->saber[0].kataMove;
|
|
}
|
|
}
|
|
if ( pm->ps->dualSabers )
|
|
{
|
|
if ( pm->ps->saber[1].kataMove != LS_INVALID )
|
|
{
|
|
if ( pm->ps->saber[1].kataMove != LS_NONE )
|
|
{
|
|
return (saberMoveName_t)pm->ps->saber[1].kataMove;
|
|
}
|
|
}
|
|
}
|
|
//no overrides, cancelled?
|
|
if ( pm->ps->saber[0].kataMove == LS_NONE )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
if ( pm->ps->dualSabers )
|
|
{
|
|
if ( pm->ps->saber[1].kataMove == LS_NONE )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
}
|
|
//do normal checks
|
|
if ( pm->ps->saberMove == LS_READY//ready
|
|
//&& (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer())//PLAYER ONLY...?
|
|
//&& pm->ps->viewangles[0] > 30 //looking down
|
|
&& pm->ps->saberAnimLevel == SS_DUAL//using dual saber style
|
|
&& pm->ps->saber[0].Active() && pm->ps->saber[1].Active()//both sabers on
|
|
//&& pm->ps->forcePowerLevel[FP_PUSH]>=FORCE_LEVEL_3//force push 3
|
|
//&& ((pm->ps->forcePowersActive&(1<<FP_PUSH))||pm->ps->forcePowerDebounce[FP_PUSH]>level.time)//force-pushing
|
|
&& 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//DUAL_SPIN_PROTECT_POWER//force push 3
|
|
&& (pm->cmd.buttons&BUTTON_ATTACK)//pressing attack
|
|
)
|
|
{//FIXME: some NPC logic to do this?
|
|
/*
|
|
if ( (pm->ps->pm_flags&PMF_DUCKED||pm->cmd.upmove<0)//crouching
|
|
&& g_crosshairEntNum >= ENTITYNUM_WORLD )
|
|
*/
|
|
{
|
|
if ( pm->gent )
|
|
{
|
|
G_DrainPowerForSpecialMove( pm->gent, FP_PUSH, SABER_ALT_ATTACK_POWER, qtrue );//drain the required force power
|
|
}
|
|
return LS_DUAL_SPIN_PROTECT;
|
|
}
|
|
}
|
|
return LS_NONE;
|
|
}
|
|
|
|
saberMoveName_t PM_CheckStaffKata( void )
|
|
{
|
|
if ( pm->ps->clientNum < MAX_CLIENTS
|
|
&& PM_InSecondaryStyle() )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
|
|
//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 )
|
|
{
|
|
return (saberMoveName_t)pm->ps->saber[0].kataMove;
|
|
}
|
|
}
|
|
if ( pm->ps->dualSabers )
|
|
{
|
|
if ( pm->ps->saber[1].kataMove != LS_INVALID )
|
|
{
|
|
if ( pm->ps->saber[1].kataMove != LS_NONE )
|
|
{
|
|
return (saberMoveName_t)pm->ps->saber[1].kataMove;
|
|
}
|
|
}
|
|
}
|
|
//no overrides, cancelled?
|
|
if ( pm->ps->saber[0].kataMove == LS_NONE )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
if ( pm->ps->dualSabers )
|
|
{
|
|
if ( pm->ps->saber[1].kataMove == LS_NONE )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
}
|
|
//do normal checks
|
|
if ( pm->ps->saberMove == LS_READY//ready
|
|
//&& (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer())//PLAYER ONLY...?
|
|
//&& pm->ps->viewangles[0] > 30 //looking down
|
|
&& pm->ps->saberAnimLevel == SS_STAFF//using dual saber style
|
|
&& pm->ps->saber[0].Active()//saber on
|
|
//&& pm->ps->forcePowerLevel[FP_PUSH]>=FORCE_LEVEL_3//force push 3
|
|
//&& ((pm->ps->forcePowersActive&(1<<FP_PUSH))||pm->ps->forcePowerDebounce[FP_PUSH]>level.time)//force-pushing
|
|
&& 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//DUAL_SPIN_PROTECT_POWER//force push 3
|
|
&& (pm->cmd.buttons&BUTTON_ATTACK)//pressing attack
|
|
)
|
|
{//FIXME: some NPC logic to do this?
|
|
/*
|
|
if ( (pm->ps->pm_flags&PMF_DUCKED||pm->cmd.upmove<0)//crouching
|
|
&& g_crosshairEntNum >= ENTITYNUM_WORLD )
|
|
*/
|
|
{
|
|
if ( pm->gent )
|
|
{
|
|
G_DrainPowerForSpecialMove( pm->gent, FP_LEVITATION, SABER_ALT_ATTACK_POWER, qtrue );//drain the required force power
|
|
}
|
|
return LS_STAFF_SOULCAL;
|
|
}
|
|
}
|
|
return LS_NONE;
|
|
}
|
|
|
|
extern qboolean WP_ForceThrowable( gentity_t *ent, gentity_t *forwardEnt, gentity_t *self, qboolean pull, float cone, float radius, vec3_t forward );
|
|
saberMoveName_t PM_CheckPullAttack( void )
|
|
{
|
|
if ( pm->ps->clientNum < MAX_CLIENTS
|
|
&& PM_InSecondaryStyle() )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
|
|
if ( (pm->ps->saber[0].saberFlags&SFL_NO_PULL_ATTACK) )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
if ( pm->ps->dualSabers
|
|
&& (pm->ps->saber[1].saberFlags&SFL_NO_PULL_ATTACK) )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
|
|
if ( (pm->ps->saberMove == LS_READY||PM_SaberInReturn(pm->ps->saberMove)||PM_SaberInReflect(pm->ps->saberMove))//ready
|
|
//&& (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer())//PLAYER ONLY
|
|
&& pm->ps->saberAnimLevel >= SS_FAST//single saber styles - FIXME: Tavion?
|
|
&& pm->ps->saberAnimLevel <= SS_STRONG//single saber styles - FIXME: Tavion?
|
|
&& G_TryingPullAttack( pm->gent, &pm->cmd, qfalse )//(pm->cmd.buttons&BUTTON_FORCE_FOCUS)//holding focus
|
|
//&& pm->cmd.forwardmove<0//pulling back
|
|
&& (pm->cmd.buttons&BUTTON_ATTACK)//attacking
|
|
&& G_EnoughPowerForSpecialMove( pm->ps->forcePower, SABER_ALT_ATTACK_POWER_FB )//pm->ps->forcePower >= SABER_ALT_ATTACK_POWER_FB//have enough power
|
|
)
|
|
{//FIXME: some NPC logic to do this?
|
|
qboolean doMove = g_saberNewControlScheme->integer?qtrue:qfalse;//in new control scheme, can always do this, even if there's no-one to do it to
|
|
if ( g_saberNewControlScheme->integer
|
|
|| g_crosshairEntNum < ENTITYNUM_WORLD )//in old control scheme, there has to be someone there
|
|
{
|
|
saberMoveName_t pullAttackMove = LS_NONE;
|
|
if ( pm->ps->saberAnimLevel == SS_FAST )
|
|
{
|
|
pullAttackMove = LS_PULL_ATTACK_STAB;
|
|
}
|
|
else
|
|
{
|
|
pullAttackMove = LS_PULL_ATTACK_SWING;
|
|
}
|
|
|
|
if ( g_crosshairEntNum < ENTITYNUM_WORLD
|
|
&& pm->gent && pm->gent->client )
|
|
{
|
|
gentity_t *targEnt = &g_entities[g_crosshairEntNum];
|
|
if ( targEnt->client
|
|
&& targEnt->health > 0
|
|
//FIXME: check other things like in knockdown, saberlock, uninterruptable anims, etc.
|
|
&& !PM_InOnGroundAnim( &targEnt->client->ps )
|
|
&& !PM_LockedAnim( targEnt->client->ps.legsAnim )
|
|
&& !PM_SuperBreakLoseAnim( targEnt->client->ps.legsAnim )
|
|
&& !PM_SuperBreakWinAnim( targEnt->client->ps.legsAnim )
|
|
&& targEnt->client->ps.saberLockTime <= 0
|
|
&& WP_ForceThrowable( targEnt, targEnt, pm->gent, qtrue, 1.0f, 0.0f, NULL ) )
|
|
{
|
|
if ( !g_saberNewControlScheme->integer )
|
|
{//in old control scheme, make sure they're close or far enough away for the move we'll be doing
|
|
float targDist = Distance( targEnt->currentOrigin, pm->ps->origin );
|
|
if ( pullAttackMove == LS_PULL_ATTACK_STAB )
|
|
{//must be closer than 512
|
|
if ( targDist > 384.0f )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
}
|
|
else//if ( pullAttackMove == LS_PULL_ATTACK_SWING )
|
|
{//must be farther than 256
|
|
if ( targDist > 512.0f )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
if ( targDist < 192.0f )
|
|
{
|
|
return LS_NONE;
|
|
}
|
|
}
|
|
}
|
|
|
|
vec3_t targAngles = {0,targEnt->client->ps.viewangles[YAW],0};
|
|
if ( InFront( pm->ps->origin, targEnt->currentOrigin, targAngles ) )
|
|
{
|
|
NPC_SetAnim( targEnt, SETANIM_BOTH, BOTH_PULLED_INAIR_F, SETANIM_FLAG_OVERRIDE, SETANIM_FLAG_HOLD );
|
|
}
|
|
else
|
|
{
|
|
NPC_SetAnim( targEnt, SETANIM_BOTH, BOTH_PULLED_INAIR_B, SETANIM_FLAG_OVERRIDE, SETANIM_FLAG_HOLD );
|
|
}
|
|
//hold the anim until I'm with done pull anim
|
|
targEnt->client->ps.legsAnimTimer = targEnt->client->ps.torsoAnimTimer = PM_AnimLength( pm->gent->client->clientInfo.animFileIndex, (animNumber_t)saberMoveData[pullAttackMove].animToUse );
|
|
//set pullAttackTime
|
|
pm->gent->client->ps.pullAttackTime = targEnt->client->ps.pullAttackTime = level.time+targEnt->client->ps.legsAnimTimer;
|
|
//make us know about each other
|
|
pm->gent->client->ps.pullAttackEntNum = g_crosshairEntNum;
|
|
targEnt->client->ps.pullAttackEntNum = pm->ps->clientNum;
|
|
//do effect and sound on me
|
|
pm->ps->powerups[PW_FORCE_PUSH] = level.time + 1000;
|
|
if ( pm->gent )
|
|
{
|
|
G_Sound( pm->gent, G_SoundIndex( "sound/weapons/force/pull.wav" ) );
|
|
}
|
|
doMove = qtrue;
|
|
}
|
|
}
|
|
if ( doMove )
|
|
{
|
|
if ( pm->gent )
|
|
{
|
|
G_DrainPowerForSpecialMove( pm->gent, FP_PULL, SABER_ALT_ATTACK_POWER_FB );
|
|
}
|
|
return pullAttackMove;
|
|
}
|
|
}
|
|
}
|
|
return LS_NONE;
|
|
}
|
|
|
|
saberMoveName_t PM_CheckPlayerAttackFromParry( int curmove )
|
|
{
|
|
if ( pm->ps->clientNum < MAX_CLIENTS || PM_ControlledByPlayer() )
|
|
{
|
|
if ( curmove >= LS_PARRY_UP
|
|
&& curmove <= LS_REFLECT_LL )
|
|
{//in a parry
|
|
switch ( saberMoveData[curmove].endQuad )
|
|
{
|
|
case Q_T:
|
|
return LS_A_T2B;
|
|
break;
|
|
case Q_TR:
|
|
return LS_A_TR2BL;
|
|
break;
|
|
case Q_TL:
|
|
return LS_A_TL2BR;
|
|
break;
|
|
case Q_BR:
|
|
return LS_A_BR2TL;
|
|
break;
|
|
case Q_BL:
|
|
return LS_A_BL2TR;
|
|
break;
|
|
//shouldn't be a parry that ends at L, R or B
|
|
}
|
|
}
|
|
}
|
|
return LS_NONE;
|
|
}
|
|
|
|
|
|
saberMoveName_t PM_SaberAttackForMovement( int forwardmove, int rightmove, int curmove )
|
|
{
|
|
qboolean noSpecials = qfalse;
|
|
|
|
if ( pm->ps->clientNum < MAX_CLIENTS
|
|
&& PM_InSecondaryStyle() )
|
|
{
|
|
noSpecials = qtrue;
|
|
}
|
|
|
|
saberMoveName_t overrideJumpRightAttackMove = LS_INVALID;
|
|
if ( pm->ps->saber[0].jumpAtkRightMove != LS_INVALID )
|
|
{
|
|
if ( pm->ps->saber[0].jumpAtkRightMove != LS_NONE )
|
|
{//actually overriding
|
|
overrideJumpRightAttackMove = (saberMoveName_t)pm->ps->saber[0].jumpAtkRightMove;
|
|
}
|
|
else if ( pm->ps->dualSabers
|
|
&& pm->ps->saber[1].jumpAtkRightMove > LS_NONE )
|
|
{//would be cancelling it, but check the second saber, too
|
|
overrideJumpRightAttackMove = (saberMoveName_t)pm->ps->saber[1].jumpAtkRightMove;
|
|
}
|
|
else
|
|
{//nope, just cancel it
|
|
overrideJumpRightAttackMove = LS_NONE;
|
|
}
|
|
}
|
|
else if ( pm->ps->dualSabers
|
|
&& pm->ps->saber[1].jumpAtkRightMove != LS_INVALID )
|
|
{//first saber not overridden, check second
|
|
overrideJumpRightAttackMove = (saberMoveName_t)pm->ps->saber[1].jumpAtkRightMove;
|
|
}
|
|
|
|
saberMoveName_t overrideJumpLeftAttackMove = LS_INVALID;
|
|
if ( pm->ps->saber[0].jumpAtkLeftMove != LS_INVALID )
|
|
{
|
|
if ( pm->ps->saber[0].jumpAtkLeftMove != LS_NONE )
|
|
{//actually overriding
|
|
overrideJumpLeftAttackMove = (saberMoveName_t)pm->ps->saber[0].jumpAtkLeftMove;
|
|
}
|
|
else if ( pm->ps->dualSabers
|
|
&& pm->ps->saber[1].jumpAtkLeftMove > LS_NONE )
|
|
{//would be cancelling it, but check the second saber, too
|
|
overrideJumpLeftAttackMove = (saberMoveName_t)pm->ps->saber[1].jumpAtkLeftMove;
|
|
}
|
|
else
|
|
{//nope, just cancel it
|
|
overrideJumpLeftAttackMove = LS_NONE;
|
|
}
|
|
}
|
|
else if ( pm->ps->dualSabers
|
|
&& pm->ps->saber[1].jumpAtkLeftMove != LS_INVALID )
|
|
{//first saber not overridden, check second
|
|
overrideJumpLeftAttackMove = (saberMoveName_t)pm->ps->saber[1].jumpAtkLeftMove;
|
|
}
|
|
#ifdef _XBOX
|
|
if ( rightmove > 64 )
|
|
#else
|
|
if ( rightmove > 0 )
|
|
#endif // _XBOX
|
|
{//moving right
|
|
if ( !noSpecials
|
|
&& overrideJumpRightAttackMove != LS_NONE
|
|
&& (pm->ps->groundEntityNum != ENTITYNUM_NONE||level.time-pm->ps->lastOnGround<=250) //on ground or just jumped
|
|
&& (pm->cmd.buttons&BUTTON_ATTACK)//hitting attack
|
|
&& pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_0//have force jump 1 at least
|
|
&& G_EnoughPowerForSpecialMove( pm->ps->forcePower, SABER_ALT_ATTACK_POWER_LR )//pm->ps->forcePower >= SABER_ALT_ATTACK_POWER_LR//have enough power
|
|
&& (((pm->ps->clientNum>=MAX_CLIENTS&&!PM_ControlledByPlayer())&&pm->cmd.upmove > 0)//jumping NPC
|
|
||((pm->ps->clientNum<MAX_CLIENTS||PM_ControlledByPlayer())&&G_TryingCartwheel(pm->gent, &pm->cmd)/*(pm->cmd.buttons&BUTTON_FORCE_FOCUS)*/)) )//focus-holding player
|
|
{//cartwheel right
|
|
vec3_t right, fwdAngles = {0, pm->ps->viewangles[YAW], 0};
|
|
if ( pm->gent )
|
|
{
|
|
G_DrainPowerForSpecialMove( pm->gent, FP_LEVITATION, SABER_ALT_ATTACK_POWER_LR );
|
|
}
|
|
pm->cmd.upmove = 0;
|
|
if ( overrideJumpRightAttackMove != LS_INVALID )
|
|
{//overridden with another move
|
|
return overrideJumpRightAttackMove;
|
|
}
|
|
else if ( pm->ps->saberAnimLevel == SS_STAFF )
|
|
{
|
|
AngleVectors( fwdAngles, NULL, right, NULL );
|
|
pm->ps->velocity[0] = pm->ps->velocity[1] = 0;
|
|
VectorMA( pm->ps->velocity, 190, right, pm->ps->velocity );
|
|
return LS_BUTTERFLY_RIGHT;
|
|
}
|
|
else
|
|
{
|
|
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->groundEntityNum != ENTITYNUM_NONE )
|
|
{//still on ground
|
|
VectorClear( pm->ps->velocity );
|
|
return LS_JUMPATTACK_CART_RIGHT;
|
|
}
|
|
else
|
|
*/
|
|
{//in air
|
|
AngleVectors( fwdAngles, NULL, right, NULL );
|
|
pm->ps->velocity[0] = pm->ps->velocity[1] = 0;
|
|
VectorMA( pm->ps->velocity, 190, right, pm->ps->velocity );
|
|
PM_SetJumped( JUMP_VELOCITY, qtrue );
|
|
return LS_JUMPATTACK_ARIAL_RIGHT;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if ( pm->ps->legsAnim != BOTH_CARTWHEEL_RIGHT
|
|
&& pm->ps->legsAnim != BOTH_ARIAL_RIGHT )
|
|
{//not in a cartwheel/arial
|
|
if ( pm->ps->clientNum < MAX_CLIENTS || PM_ControlledByPlayer() ) //PLAYER ONLY
|
|
{//player
|
|
if ( G_TryingSpecial(pm->gent, &pm->cmd)/*(pm->cmd.buttons&BUTTON_FORCE_FOCUS)*/ )//Holding focus
|
|
{//if no special worked, do nothing
|
|
return LS_NONE;
|
|
}
|
|
}
|
|
//checked all special attacks, if we're in a parry, attack from that move
|
|
saberMoveName_t parryAttackMove = PM_CheckPlayerAttackFromParry( curmove );
|
|
if ( parryAttackMove != LS_NONE )
|
|
{
|
|
return parryAttackMove;
|
|
}
|
|
//check regular attacks
|
|
if ( forwardmove > 0 )
|
|
{//forward right = TL2BR slash
|
|
return LS_A_TL2BR;
|
|
}
|
|
else if ( forwardmove < 0 )
|
|
{//backward right = BL2TR uppercut
|
|
return LS_A_BL2TR;
|
|
}
|
|
else
|
|
{//just right is a left slice
|
|
return LS_A_L2R;
|
|
}
|
|
}
|
|
}
|
|
#ifdef _XBOX
|
|
else if ( rightmove < -64 )
|
|
#else
|
|
else if ( rightmove < 0 )
|
|
#endif // _XBOX
|
|
{//moving left
|
|
if ( !noSpecials
|
|
&& overrideJumpLeftAttackMove != LS_NONE
|
|
&& (pm->ps->groundEntityNum != ENTITYNUM_NONE||level.time-pm->ps->lastOnGround<=250) //on ground or just jumped
|
|
&& (pm->cmd.buttons&BUTTON_ATTACK)//hitting attack
|
|
&& pm->ps->forcePowerLevel[FP_LEVITATION] > FORCE_LEVEL_0//have force jump 1 at least
|
|
&& G_EnoughPowerForSpecialMove( pm->ps->forcePower, SABER_ALT_ATTACK_POWER_LR )//pm->ps->forcePower >= SABER_ALT_ATTACK_POWER_LR//have enough power
|
|
&& (((pm->ps->clientNum>=MAX_CLIENTS&&!PM_ControlledByPlayer())&&pm->cmd.upmove > 0)//jumping NPC
|
|
||((pm->ps->clientNum<MAX_CLIENTS||PM_ControlledByPlayer())&&G_TryingCartwheel(pm->gent, &pm->cmd)/*(pm->cmd.buttons&BUTTON_FORCE_FOCUS)*/)) )//focus-holding player
|
|
{//cartwheel left
|
|
vec3_t right, fwdAngles = {0, pm->ps->viewangles[YAW], 0};
|
|
if ( pm->gent )
|
|
{
|
|
G_DrainPowerForSpecialMove( pm->gent, FP_LEVITATION, SABER_ALT_ATTACK_POWER_LR );
|
|
}
|
|
pm->cmd.upmove = 0;
|
|
if ( overrideJumpRightAttackMove != LS_INVALID )
|
|
{//overridden with another move
|
|
return overrideJumpRightAttackMove;
|
|
}
|
|
else if ( pm->ps->saberAnimLevel == SS_STAFF )
|
|
{
|
|
AngleVectors( fwdAngles, NULL, right, NULL );
|
|
pm->ps->velocity[0] = pm->ps->velocity[1] = 0;
|
|
VectorMA( pm->ps->velocity, -190, right, pm->ps->velocity );
|
|
return LS_BUTTERFLY_LEFT;
|
|
}
|
|
else
|
|
{
|
|
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->groundEntityNum != ENTITYNUM_NONE )
|
|
{//still on ground
|
|
VectorClear( pm->ps->velocity );
|
|
return LS_JUMPATTACK_ARIAL_LEFT;
|
|
}
|
|
else
|
|
*/
|
|
{
|
|
AngleVectors( fwdAngles, NULL, right, NULL );
|
|
pm->ps->velocity[0] = pm->ps->velocity[1] = 0;
|
|
VectorMA( pm->ps->velocity, -190, right, pm->ps->velocity );
|
|
PM_SetJumped( JUMP_VELOCITY, qtrue );
|
|
return LS_JUMPATTACK_CART_LEFT;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if ( pm->ps->legsAnim != BOTH_CARTWHEEL_LEFT
|
|
&& pm->ps->legsAnim != BOTH_ARIAL_LEFT )
|
|
{//not in a left cartwheel/arial
|
|
if ( pm->ps->clientNum < MAX_CLIENTS || PM_ControlledByPlayer() ) //PLAYER ONLY
|
|
{//player
|
|
if ( G_TryingSpecial(pm->gent, &pm->cmd)/*(pm->cmd.buttons&BUTTON_FORCE_FOCUS)*/ )//Holding focus
|
|
{//if no special worked, do nothing
|
|
return LS_NONE;
|
|
}
|
|
}
|
|
//checked all special attacks, if we're in a parry, attack from that move
|
|
saberMoveName_t parryAttackMove = PM_CheckPlayerAttackFromParry( curmove );
|
|
if ( parryAttackMove != LS_NONE )
|
|
{
|
|
return parryAttackMove;
|
|
}
|
|
//check regular attacks
|
|
if ( forwardmove > 0 )
|
|
{//forward left = TR2BL slash
|
|
return LS_A_TR2BL;
|
|
}
|
|
else if ( forwardmove < 0 )
|
|
{//backward left = BR2TL uppercut
|
|
return LS_A_BR2TL;
|
|
}
|
|
else
|
|
{//just left is a right slice
|
|
return LS_A_R2L;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{//not moving left or right
|
|
#ifdef _XBOX
|
|
if ( forwardmove > 64 )
|
|
#else
|
|
if ( forwardmove > 0 )
|
|
#endif // _XBOX
|
|
{//forward= T2B slash
|
|
saberMoveName_t stabDownMove = noSpecials?LS_NONE:PM_CheckStabDown();
|
|
if ( stabDownMove != LS_NONE )
|
|
{
|
|
return stabDownMove;
|
|
}
|
|
if ( ((pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && cg.renderingThirdPerson && !cg.zoomMode) )//player in third person, not zoomed in
|
|
{//player in thirdperson, not zoomed in
|
|
//flip-over attack logic
|
|
if ( !noSpecials && PM_CheckFlipOverAttackMove( qfalse ) )
|
|
{//flip over-forward down-attack
|
|
return PM_SaberFlipOverAttackMove();
|
|
}
|
|
//lunge attack logic
|
|
else if ( PM_CheckLungeAttackMove() )
|
|
{
|
|
return PM_SaberLungeAttackMove( qtrue );
|
|
}
|
|
//jump forward attack logic
|
|
else if ( !noSpecials && PM_CheckJumpForwardAttackMove() )
|
|
{
|
|
return PM_SaberJumpForwardAttackMove();
|
|
}
|
|
}
|
|
|
|
//player NPC with enemy: autoMove logic
|
|
if ( pm->gent
|
|
&& pm->gent->enemy
|
|
&& pm->gent->enemy->client )
|
|
{//I have an active enemy
|
|
if ( pm->ps->clientNum < MAX_CLIENTS || PM_ControlledByPlayer() )
|
|
{//a player who is running at an enemy
|
|
//if the enemy is not a jedi, don't use top-down, pick a diagonal or side attack
|
|
if ( pm->gent->enemy->s.weapon != WP_SABER
|
|
&& pm->gent->enemy->client->NPC_class != CLASS_REMOTE//too small to do auto-aiming accurately
|
|
&& pm->gent->enemy->client->NPC_class != CLASS_SEEKER//too small to do auto-aiming accurately
|
|
&& pm->gent->enemy->client->NPC_class != CLASS_GONK//too short to do auto-aiming accurately
|
|
&& pm->gent->enemy->client->NPC_class != CLASS_HOWLER//too short to do auto-aiming accurately
|
|
&& g_saberAutoAim->integer )
|
|
{
|
|
saberMoveName_t autoMove = PM_AttackForEnemyPos( qfalse, (qboolean)(pm->ps->clientNum>=MAX_CLIENTS&&!PM_ControlledByPlayer()) );
|
|
if ( autoMove != LS_INVALID )
|
|
{
|
|
return autoMove;
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( pm->ps->clientNum>=MAX_CLIENTS && !PM_ControlledByPlayer() ) //NPC ONLY
|
|
{//NPC
|
|
if ( PM_CheckFlipOverAttackMove( qtrue ) )
|
|
{
|
|
return PM_SaberFlipOverAttackMove();
|
|
}
|
|
}
|
|
}
|
|
|
|
//Regular NPCs
|
|
if ( pm->ps->clientNum >= MAX_CLIENTS && !PM_ControlledByPlayer() ) //NPC ONLY
|
|
{//NPC or player in third person, not zoomed in
|
|
//fwd jump attack logic
|
|
if ( PM_CheckJumpForwardAttackMove() )
|
|
{
|
|
return PM_SaberJumpForwardAttackMove();
|
|
}
|
|
//lunge attack logic
|
|
else if ( PM_CheckLungeAttackMove() )
|
|
{
|
|
return PM_SaberLungeAttackMove( qtrue );
|
|
}
|
|
}
|
|
|
|
if ( pm->ps->clientNum < MAX_CLIENTS || PM_ControlledByPlayer() ) //PLAYER ONLY
|
|
{//player
|
|
if ( G_TryingSpecial(pm->gent,&pm->cmd) )//(pm->cmd.buttons&BUTTON_FORCE_FOCUS) )//Holding focus
|
|
{//if no special worked, do nothing
|
|
return LS_NONE;
|
|
}
|
|
}
|
|
|
|
//checked all special attacks, if we're in a parry, attack from that move
|
|
saberMoveName_t parryAttackMove = PM_CheckPlayerAttackFromParry( curmove );
|
|
if ( parryAttackMove != LS_NONE )
|
|
{
|
|
return parryAttackMove;
|
|
}
|
|
//check regular attacks
|
|
return LS_A_T2B;
|
|
}
|
|
#ifdef _XBOX
|
|
else if ( forwardmove < -64 )
|
|
#else
|
|
else if ( forwardmove < 0 )
|
|
#endif // _XBOX
|
|
{//backward= T2B slash//B2T uppercut?
|
|
if ( g_saberNewControlScheme->integer )
|
|
{
|
|
saberMoveName_t pullAtk = PM_CheckPullAttack();
|
|
if ( pullAtk != LS_NONE )
|
|
{
|
|
return pullAtk;
|
|
}
|
|
}
|
|
|
|
if ( g_saberNewControlScheme->integer
|
|
&& (pm->ps->clientNum < MAX_CLIENTS || PM_ControlledByPlayer()) //PLAYER ONLY
|
|
&& (pm->cmd.buttons&BUTTON_FORCE_FOCUS) )//Holding focus, trying special backwards attacks
|
|
{//player lunge attack logic
|
|
if ( ( pm->ps->dualSabers //or dual
|
|
|| pm->ps->saberAnimLevel == SS_STAFF )//pm->ps->SaberStaff() )//or staff
|
|
&& G_EnoughPowerForSpecialMove( pm->ps->forcePower, SABER_ALT_ATTACK_POWER_FB )/*pm->ps->forcePower >= SABER_ALT_ATTACK_POWER_FB*/ )//have enough force power to pull it off
|
|
{//alt+back+attack using fast, dual or staff attacks
|
|
PM_SaberLungeAttackMove( qfalse );
|
|
}
|
|
}
|
|
else if ( (pm->ps->clientNum&&!PM_ControlledByPlayer()) //NPC
|
|
|| ((pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && cg.renderingThirdPerson && !cg.zoomMode) )//player in third person, not zooomed
|
|
{//NPC or player in third person, not zoomed
|
|
if ( PM_CheckBackflipAttackMove() )
|
|
{
|
|
return PM_SaberBackflipAttackMove();//backflip attack
|
|
}
|
|
if ( pm->ps->clientNum < MAX_CLIENTS || PM_ControlledByPlayer() ) //PLAYER ONLY
|
|
{//player
|
|
if ( G_TryingSpecial(pm->gent,&pm->cmd) )//(pm->cmd.buttons&BUTTON_FORCE_FOCUS) )//Holding focus
|
|
{//if no special worked, do nothing
|
|
return LS_NONE;
|
|
}
|
|
}
|
|
//if ( !PM_InKnockDown( pm->ps ) )
|
|
//check backstabs
|
|
if ( !(pm->ps->saber[0].saberFlags&SFL_NO_BACK_ATTACK)
|
|
&& (!pm->ps->dualSabers || !(pm->ps->saber[1].saberFlags&SFL_NO_BACK_ATTACK)) )
|
|
{//okay to do backstabs with this saber
|
|
if ( pm->ps->groundEntityNum != ENTITYNUM_NONE )
|
|
{//only when on ground
|
|
if ( pm->gent && pm->gent->enemy )
|
|
{//FIXME: or just trace for a valid enemy standing behind me? And no enemy in front?
|
|
vec3_t enemyDir, faceFwd, facingAngles = {0, pm->ps->viewangles[YAW], 0};
|
|
AngleVectors( facingAngles, faceFwd, NULL, NULL );
|
|
VectorSubtract( pm->gent->enemy->currentOrigin, pm->ps->origin, enemyDir );
|
|
float dot = DotProduct( enemyDir, faceFwd );
|
|
if ( dot < 0 )
|
|
{//enemy is behind me
|
|
if ( dot < -0.75f
|
|
&& DistanceSquared( pm->gent->currentOrigin, pm->gent->enemy->currentOrigin ) < 16384//128 squared
|
|
&& (pm->ps->saberAnimLevel == SS_FAST || pm->ps->saberAnimLevel == SS_STAFF || (pm->gent->client &&(pm->gent->client->NPC_class == CLASS_TAVION||pm->gent->client->NPC_class == CLASS_ALORA)&&Q_irand(0,1))) )
|
|
{//fast attacks and Tavion
|
|
if ( !(pm->ps->pm_flags&PMF_DUCKED) && pm->cmd.upmove >= 0 )
|
|
{//can't do it while ducked?
|
|
if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) || (pm->gent->NPC && pm->gent->NPC->rank >= RANK_LT_JG) )
|
|
{//only fencers and above can do this
|
|
return LS_A_BACKSTAB;
|
|
}
|
|
}
|
|
}
|
|
else if ( pm->ps->saberAnimLevel != SS_FAST
|
|
&& pm->ps->saberAnimLevel != SS_STAFF )
|
|
{//medium and higher attacks
|
|
if ( (pm->ps->pm_flags&PMF_DUCKED) || pm->cmd.upmove < 0 )
|
|
{
|
|
return LS_A_BACK_CR;
|
|
}
|
|
else
|
|
{
|
|
return LS_A_BACK;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{//enemy in front
|
|
float enemyDistSq = DistanceSquared( pm->gent->currentOrigin, pm->gent->enemy->currentOrigin );
|
|
if ( (pm->ps->saberAnimLevel == FORCE_LEVEL_1||pm->ps->saberAnimLevel == SS_STAFF||pm->gent->client->NPC_class==CLASS_TAVION||pm->gent->client->NPC_class == CLASS_ALORA||(pm->gent->client->NPC_class==CLASS_DESANN&&!Q_irand(0,3))) && enemyDistSq > 16384 || pm->gent->enemy->health <= 0 )//128 squared
|
|
{//my enemy is pretty far in front of me and I'm using fast attacks
|
|
if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) ||
|
|
( pm->gent && pm->gent->client && pm->gent->NPC && pm->gent->NPC->rank >= RANK_LT_JG && Q_irand( 0, pm->gent->NPC->rank ) > RANK_ENSIGN ) )
|
|
{//only fencers and higher can do this, higher rank does it more
|
|
if ( PM_CheckEnemyInBack( 128 ) )
|
|
{
|
|
return PM_PickBackStab();
|
|
}
|
|
}
|
|
}
|
|
else if ( (pm->ps->saberAnimLevel >= FORCE_LEVEL_2 || pm->gent->client->NPC_class == CLASS_DESANN) && enemyDistSq > 40000 || pm->gent->enemy->health <= 0 )//200 squared
|
|
{//enemy is very faw away and I'm using medium/strong attacks
|
|
if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) ||
|
|
( pm->gent && pm->gent->client && pm->gent->NPC && pm->gent->NPC->rank >= RANK_LT_JG && Q_irand( 0, pm->gent->NPC->rank ) > RANK_ENSIGN ) )
|
|
{//only fencers and higher can do this, higher rank does it more
|
|
if ( PM_CheckEnemyInBack( 164 ) )
|
|
{
|
|
return PM_PickBackStab();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{//no current enemy
|
|
if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && pm->gent && pm->gent->client )
|
|
{//only player
|
|
if ( PM_CheckEnemyInBack( 128 ) )
|
|
{
|
|
return PM_PickBackStab();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( pm->ps->clientNum < MAX_CLIENTS || PM_ControlledByPlayer() ) //PLAYER ONLY
|
|
{//player
|
|
if ( G_TryingSpecial( pm->gent, &pm->cmd ) )//(pm->cmd.buttons&BUTTON_FORCE_FOCUS) )//Holding focus
|
|
{//if no special worked, do nothing
|
|
return LS_NONE;
|
|
}
|
|
}
|
|
|
|
//checked all special attacks, if we're in a parry, attack from that move
|
|
saberMoveName_t parryAttackMove = PM_CheckPlayerAttackFromParry( curmove );
|
|
if ( parryAttackMove != LS_NONE )
|
|
{
|
|
return parryAttackMove;
|
|
}
|
|
//check regular attacks
|
|
//else just swing down
|
|
return LS_A_T2B;
|
|
}
|
|
else
|
|
{//not moving in any direction
|
|
if ( PM_SaberInBounce( curmove ) )
|
|
{//bounces should go to their default attack if you don't specify a direction but are attacking
|
|
if ( pm->ps->clientNum < MAX_CLIENTS || PM_ControlledByPlayer() ) //PLAYER ONLY
|
|
{//player
|
|
if ( G_TryingSpecial(pm->gent,&pm->cmd) )//(pm->cmd.buttons&BUTTON_FORCE_FOCUS) )//Holding focus
|
|
{//if no special worked, do nothing
|
|
return LS_NONE;
|
|
}
|
|
}
|
|
saberMoveName_t newmove;
|
|
if ( pm->ps->clientNum && !PM_ControlledByPlayer() && Q_irand( 0, 3 ) )
|
|
{//use NPC random
|
|
newmove = PM_NPCSaberAttackFromQuad( saberMoveData[curmove].endQuad );
|
|
}
|
|
else
|
|
{//player uses chain-attack
|
|
newmove = saberMoveData[curmove].chain_attack;
|
|
}
|
|
if ( PM_SaberKataDone( curmove, newmove ) )
|
|
{
|
|
return saberMoveData[curmove].chain_idle;
|
|
}
|
|
else
|
|
{
|
|
return newmove;
|
|
}
|
|
}
|
|
else if ( PM_SaberInKnockaway( curmove ) )
|
|
{//bounces should go to their default attack if you don't specify a direction but are attacking
|
|
if ( pm->ps->clientNum < MAX_CLIENTS || PM_ControlledByPlayer() ) //PLAYER ONLY
|
|
{//player
|
|
if ( G_TryingSpecial( pm->gent, &pm->cmd ) )//(pm->cmd.buttons&BUTTON_FORCE_FOCUS) )//Holding focus
|
|
{//if no special worked, do nothing
|
|
return LS_NONE;
|
|
}
|
|
}
|
|
saberMoveName_t newmove;
|
|
if ( pm->ps->clientNum && !PM_ControlledByPlayer() && Q_irand( 0, 3 ) )
|
|
{//use NPC random
|
|
newmove = PM_NPCSaberAttackFromQuad( saberMoveData[curmove].endQuad );
|
|
}
|
|
else
|
|
{
|
|
if ( pm->ps->saberAnimLevel == SS_FAST ||
|
|
pm->ps->saberAnimLevel == SS_TAVION )
|
|
{//player is in fast attacks, so come right back down from the same spot
|
|
newmove = PM_AttackMoveForQuad( saberMoveData[curmove].endQuad );
|
|
}
|
|
else
|
|
{//use a transition to wrap to another attack from a different dir
|
|
newmove = saberMoveData[curmove].chain_attack;
|
|
}
|
|
}
|
|
if ( PM_SaberKataDone( curmove, newmove ) )
|
|
{
|
|
return saberMoveData[curmove].chain_idle;
|
|
}
|
|
else
|
|
{
|
|
return newmove;
|
|
}
|
|
}
|
|
else if ( curmove == LS_READY
|
|
|| curmove == LS_A_FLIP_STAB
|
|
|| curmove == LS_A_FLIP_SLASH
|
|
|| ( curmove >= LS_PARRY_UP
|
|
&& curmove <= LS_REFLECT_LL ) )
|
|
{//Not moving at all, not too busy to attack
|
|
//push + lookdown + attack + dual sabers = LS_DUAL_SPIN_PROTECT
|
|
if ( g_saberNewControlScheme->integer )
|
|
{
|
|
if ( PM_CheckDualSpinProtect() )
|
|
{
|
|
return LS_DUAL_SPIN_PROTECT;
|
|
}
|
|
if ( PM_CheckStaffKata() )
|
|
{
|
|
return LS_STAFF_SOULCAL;
|
|
}
|
|
}
|
|
if ( pm->ps->clientNum < MAX_CLIENTS || PM_ControlledByPlayer() ) //PLAYER ONLY
|
|
{//player
|
|
if ( G_TryingSpecial( pm->gent, &pm->cmd ) )//(pm->cmd.buttons&BUTTON_FORCE_FOCUS) )//Holding focus
|
|
{//if no special worked, do nothing
|
|
return LS_NONE;
|
|
}
|
|
}
|
|
//checked all special attacks, if we're in a parry, attack from that move
|
|
saberMoveName_t parryAttackMove = PM_CheckPlayerAttackFromParry( curmove );
|
|
if ( parryAttackMove != LS_NONE )
|
|
{
|
|
return parryAttackMove;
|
|
}
|
|
//check regular attacks
|
|
if ( pm->ps->clientNum || g_saberAutoAim->integer )
|
|
{//auto-aim
|
|
if ( pm->gent && pm->gent->enemy )
|
|
{//based on enemy position, pick a proper attack
|
|
saberMoveName_t autoMove = PM_AttackForEnemyPos( qtrue, (qboolean)(pm->ps->clientNum>=MAX_CLIENTS) );
|
|
if ( autoMove != LS_INVALID )
|
|
{
|
|
return autoMove;
|
|
}
|
|
}
|
|
else if ( fabs(pm->ps->viewangles[0]) > 30 )
|
|
{//looking far up or far down uses the top to bottom attack, presuming you want a vertical attack
|
|
return LS_A_T2B;
|
|
}
|
|
}
|
|
else
|
|
{//for now, just pick a random attack
|
|
return ((saberMoveName_t)Q_irand( LS_A_TL2BR, LS_A_T2B ));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
//FIXME: pick a return?
|
|
return LS_NONE;
|
|
}
|
|
|
|
saberMoveName_t PM_SaberAnimTransitionMove( saberMoveName_t curmove, saberMoveName_t newmove )
|
|
{
|
|
//FIXME: take FP_SABER_OFFENSE into account here somehow?
|
|
int retmove = newmove;
|
|
if ( curmove == LS_READY )
|
|
{//just standing there
|
|
switch ( newmove )
|
|
{
|
|
case LS_A_TL2BR:
|
|
case LS_A_L2R:
|
|
case LS_A_BL2TR:
|
|
case LS_A_BR2TL:
|
|
case LS_A_R2L:
|
|
case LS_A_TR2BL:
|
|
case LS_A_T2B:
|
|
//transition is the start
|
|
retmove = LS_S_TL2BR + (newmove-LS_A_TL2BR);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
switch ( newmove )
|
|
{
|
|
//transitioning to ready pose
|
|
case LS_READY:
|
|
switch ( curmove )
|
|
{
|
|
//transitioning from an attack
|
|
case LS_A_TL2BR:
|
|
case LS_A_L2R:
|
|
case LS_A_BL2TR:
|
|
case LS_A_BR2TL:
|
|
case LS_A_R2L:
|
|
case LS_A_TR2BL:
|
|
case LS_A_T2B:
|
|
//transition is the return
|
|
retmove = LS_R_TL2BR + (newmove-LS_A_TL2BR);
|
|
break;
|
|
}
|
|
break;
|
|
//transitioning to an attack
|
|
case LS_A_TL2BR:
|
|
case LS_A_L2R:
|
|
case LS_A_BL2TR:
|
|
case LS_A_BR2TL:
|
|
case LS_A_R2L:
|
|
case LS_A_TR2BL:
|
|
case LS_A_T2B:
|
|
if ( newmove == curmove )
|
|
{//FIXME: need a spin or something or go to next level, but for now, just play the return
|
|
//going into another attack...
|
|
//allow endless chaining in level 1 attacks, several in level 2 and only one or a few in level 3
|
|
//FIXME: don't let strong attacks chain to an attack in the opposite direction ( > 45 degrees?)
|
|
if ( PM_SaberKataDone( curmove, newmove ) )
|
|
{//done with this kata, must return to ready before attack again
|
|
retmove = LS_R_TL2BR + (newmove-LS_A_TL2BR);
|
|
}
|
|
else
|
|
{//okay to chain to another attack
|
|
retmove = transitionMove[saberMoveData[curmove].endQuad][saberMoveData[newmove].startQuad];
|
|
}
|
|
}
|
|
else if ( saberMoveData[curmove].endQuad == saberMoveData[newmove].startQuad )
|
|
{//new move starts from same quadrant
|
|
retmove = newmove;
|
|
}
|
|
else
|
|
{
|
|
switch ( curmove )
|
|
{
|
|
//transitioning from an attack
|
|
case LS_A_TL2BR:
|
|
case LS_A_L2R:
|
|
case LS_A_BL2TR:
|
|
case LS_A_BR2TL:
|
|
case LS_A_R2L:
|
|
case LS_A_TR2BL:
|
|
case LS_A_T2B:
|
|
case LS_D1_BR:
|
|
case LS_D1__R:
|
|
case LS_D1_TR:
|
|
case LS_D1_T_:
|
|
case LS_D1_TL:
|
|
case LS_D1__L:
|
|
case LS_D1_BL:
|
|
case LS_D1_B_:
|
|
retmove = transitionMove[saberMoveData[curmove].endQuad][saberMoveData[newmove].startQuad];
|
|
break;
|
|
//transitioning from a return
|
|
case LS_R_TL2BR:
|
|
case LS_R_L2R:
|
|
case LS_R_BL2TR:
|
|
case LS_R_BR2TL:
|
|
case LS_R_R2L:
|
|
case LS_R_TR2BL:
|
|
case LS_R_T2B:
|
|
//transitioning from a bounce
|
|
/*
|
|
case LS_BOUNCE_UL2LL:
|
|
case LS_BOUNCE_LL2UL:
|
|
case LS_BOUNCE_L2LL:
|
|
case LS_BOUNCE_L2UL:
|
|
case LS_BOUNCE_UR2LR:
|
|
case LS_BOUNCE_LR2UR:
|
|
case LS_BOUNCE_R2LR:
|
|
case LS_BOUNCE_R2UR:
|
|
case LS_BOUNCE_TOP:
|
|
case LS_OVER_UR2UL:
|
|
case LS_OVER_UL2UR:
|
|
case LS_BOUNCE_UR:
|
|
case LS_BOUNCE_UL:
|
|
case LS_BOUNCE_LR:
|
|
case LS_BOUNCE_LL:
|
|
*/
|
|
//transitioning from a parry/reflection/knockaway/broken parry
|
|
case LS_PARRY_UP:
|
|
case LS_PARRY_UR:
|
|
case LS_PARRY_UL:
|
|
case LS_PARRY_LR:
|
|
case LS_PARRY_LL:
|
|
case LS_REFLECT_UP:
|
|
case LS_REFLECT_UR:
|
|
case LS_REFLECT_UL:
|
|
case LS_REFLECT_LR:
|
|
case LS_REFLECT_LL:
|
|
case LS_K1_T_:
|
|
case LS_K1_TR:
|
|
case LS_K1_TL:
|
|
case LS_K1_BR:
|
|
case LS_K1_BL:
|
|
case LS_V1_BR:
|
|
case LS_V1__R:
|
|
case LS_V1_TR:
|
|
case LS_V1_T_:
|
|
case LS_V1_TL:
|
|
case LS_V1__L:
|
|
case LS_V1_BL:
|
|
case LS_V1_B_:
|
|
case LS_H1_T_:
|
|
case LS_H1_TR:
|
|
case LS_H1_TL:
|
|
case LS_H1_BR:
|
|
case LS_H1_BL:
|
|
retmove = transitionMove[saberMoveData[curmove].endQuad][saberMoveData[newmove].startQuad];
|
|
break;
|
|
//NB: transitioning from transitions is fine
|
|
}
|
|
}
|
|
break;
|
|
//transitioning to any other anim is not supported
|
|
}
|
|
}
|
|
|
|
if ( retmove == LS_NONE )
|
|
{
|
|
return newmove;
|
|
}
|
|
|
|
return ((saberMoveName_t)retmove);
|
|
}
|
|
|
|
/*
|
|
-------------------------
|
|
PM_LegsAnimForFrame
|
|
Returns animNumber for current frame
|
|
-------------------------
|
|
*/
|
|
int PM_LegsAnimForFrame( gentity_t *ent, int legsFrame )
|
|
{
|
|
//Must be a valid client
|
|
if ( ent->client == NULL )
|
|
return -1;
|
|
|
|
//Must have a file index entry
|
|
if( ValidAnimFileIndex( ent->client->clientInfo.animFileIndex ) == qfalse )
|
|
return -1;
|
|
|
|
animation_t *animations = level.knownAnimFileSets[ent->client->clientInfo.animFileIndex].animations;
|
|
int glaIndex = gi.G2API_GetAnimIndex(&(ent->ghoul2[0]));
|
|
|
|
for ( int animation = 0; animation < BOTH_CIN_1; animation++ ) //first anim after last legs
|
|
{
|
|
if ( animation >= TORSO_DROPWEAP1 && animation < LEGS_TURN1 ) //first legs only anim
|
|
{//not a possible legs anim
|
|
continue;
|
|
}
|
|
|
|
if ( animations[animation].glaIndex != glaIndex )
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if ( animations[animation].firstFrame > legsFrame )
|
|
{//This anim starts after this frame
|
|
continue;
|
|
}
|
|
|
|
if ( animations[animation].firstFrame + animations[animation].numFrames < legsFrame )
|
|
{//This anim ends before this frame
|
|
continue;
|
|
}
|
|
//else, must be in this anim!
|
|
return animation;
|
|
}
|
|
|
|
//Not in ANY torsoAnim? SHOULD NEVER HAPPEN
|
|
// assert(0);
|
|
return -1;
|
|
}
|
|
|
|
int PM_ValidateAnimRange( const int startFrame, const int endFrame, const float animSpeed )
|
|
{//given a startframe and endframe, see if that lines up with any known animation
|
|
animation_t *animations = level.knownAnimFileSets[0].animations;
|
|
|
|
for ( int anim = 0; anim < MAX_ANIMATIONS; anim++ )
|
|
{
|
|
if ( animSpeed < 0 )
|
|
{//playing backwards
|
|
if ( animations[anim].firstFrame == endFrame )
|
|
{
|
|
if ( animations[anim].numFrames + animations[anim].firstFrame == startFrame )
|
|
{
|
|
//Com_Printf( "valid reverse anim: %s\n", animTable[anim].name );
|
|
return anim;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{//playing forwards
|
|
if ( animations[anim].firstFrame == startFrame )
|
|
{//This anim starts on this frame
|
|
if ( animations[anim].firstFrame + animations[anim].numFrames == endFrame )
|
|
{//This anim ends on this frame
|
|
//Com_Printf( "valid forward anim: %s\n", animTable[anim].name );
|
|
return anim;
|
|
}
|
|
}
|
|
}
|
|
//else, must not be this anim!
|
|
}
|
|
|
|
//Not in ANY anim? SHOULD NEVER HAPPEN
|
|
Com_Printf( "invalid anim range %d to %d, speed %4.2f\n", startFrame, endFrame, animSpeed );
|
|
return -1;
|
|
}
|
|
/*
|
|
-------------------------
|
|
PM_TorsoAnimForFrame
|
|
Returns animNumber for current frame
|
|
-------------------------
|
|
*/
|
|
int PM_TorsoAnimForFrame( gentity_t *ent, int torsoFrame )
|
|
{
|
|
//Must be a valid client
|
|
if ( ent->client == NULL )
|
|
return -1;
|
|
|
|
//Must have a file index entry
|
|
if( ValidAnimFileIndex( ent->client->clientInfo.animFileIndex ) == qfalse )
|
|
return -1;
|
|
|
|
animation_t *animations = level.knownAnimFileSets[ent->client->clientInfo.animFileIndex].animations;
|
|
int glaIndex = gi.G2API_GetAnimIndex(&(ent->ghoul2[0]));
|
|
|
|
for ( int animation = 0; animation < LEGS_TURN1; animation++ ) //first legs only anim
|
|
{
|
|
if ( animations[animation].glaIndex != glaIndex )
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if ( animations[animation].firstFrame > torsoFrame )
|
|
{//This anim starts after this frame
|
|
continue;
|
|
}
|
|
|
|
if ( animations[animation].firstFrame + animations[animation].numFrames < torsoFrame )
|
|
{//This anim ends before this frame
|
|
continue;
|
|
}
|
|
//else, must be in this anim!
|
|
return animation;
|
|
}
|
|
|
|
//Not in ANY torsoAnim? SHOULD NEVER HAPPEN
|
|
// assert(0);
|
|
return -1;
|
|
}
|
|
|
|
qboolean PM_FinishedCurrentLegsAnim( gentity_t *self )
|
|
{
|
|
int junk, curFrame;
|
|
float currentFrame, animSpeed;
|
|
|
|
if ( !self->client )
|
|
{
|
|
return qtrue;
|
|
}
|
|
|
|
gi.G2API_GetBoneAnimIndex( &self->ghoul2[self->playerModel], self->rootBone, (cg.time?cg.time:level.time), ¤tFrame, &junk, &junk, &junk, &animSpeed, NULL );
|
|
curFrame = floor( currentFrame );
|
|
|
|
int legsAnim = self->client->ps.legsAnim;
|
|
animation_t *animations = level.knownAnimFileSets[self->client->clientInfo.animFileIndex].animations;
|
|
|
|
if ( curFrame >= animations[legsAnim].firstFrame + (animations[legsAnim].numFrames - 2) )
|
|
{
|
|
return qtrue;
|
|
}
|
|
|
|
return qfalse;
|
|
}
|
|
|
|
/*
|
|
-------------------------
|
|
PM_HasAnimation
|
|
-------------------------
|
|
*/
|
|
|
|
qboolean PM_HasAnimation( gentity_t *ent, int animation )
|
|
{
|
|
//Must be a valid client
|
|
if ( !ent || ent->client == NULL )
|
|
return qfalse;
|
|
|
|
//must be a valid anim number
|
|
if ( animation < 0 || animation >= MAX_ANIMATIONS )
|
|
{
|
|
return qfalse;
|
|
}
|
|
//Must have a file index entry
|
|
if( ValidAnimFileIndex( ent->client->clientInfo.animFileIndex ) == qfalse )
|
|
return qfalse;
|
|
|
|
animation_t *animations = level.knownAnimFileSets[ent->client->clientInfo.animFileIndex].animations;
|
|
|
|
//No frames, no anim
|
|
if ( animations[animation].numFrames == 0 )
|
|
return qfalse;
|
|
|
|
//Has the sequence
|
|
return qtrue;
|
|
}
|
|
|
|
int PM_PickAnim( gentity_t *self, int minAnim, int maxAnim )
|
|
{
|
|
int anim;
|
|
int count = 0;
|
|
|
|
if ( !self )
|
|
{
|
|
return Q_irand(minAnim, maxAnim);
|
|
}
|
|
|
|
do
|
|
{
|
|
anim = Q_irand(minAnim, maxAnim);
|
|
count++;
|
|
}
|
|
while ( !PM_HasAnimation( self, anim ) && count < 1000 );
|
|
|
|
return anim;
|
|
}
|
|
|
|
/*
|
|
-------------------------
|
|
PM_AnimLength
|
|
-------------------------
|
|
*/
|
|
|
|
int PM_AnimLength( int index, animNumber_t anim )
|
|
{
|
|
if ( ValidAnimFileIndex( index ) == false )
|
|
return 0;
|
|
|
|
return level.knownAnimFileSets[index].animations[anim].numFrames * abs(level.knownAnimFileSets[index].animations[anim].frameLerp);
|
|
}
|
|
|
|
/*
|
|
-------------------------
|
|
PM_SetLegsAnimTimer
|
|
-------------------------
|
|
*/
|
|
|
|
void PM_SetLegsAnimTimer( gentity_t *ent, int *legsAnimTimer, int time )
|
|
{
|
|
*legsAnimTimer = time;
|
|
|
|
if ( *legsAnimTimer < 0 && time != -1 )
|
|
{//Cap timer to 0 if was counting down, but let it be -1 if that was intentional
|
|
*legsAnimTimer = 0;
|
|
}
|
|
|
|
if ( !*legsAnimTimer && ent && Q3_TaskIDPending( ent, TID_ANIM_LOWER ) )
|
|
{//Waiting for legsAnimTimer to complete, and it just got set to zero
|
|
if ( !Q3_TaskIDPending( ent, TID_ANIM_BOTH) )
|
|
{//Not waiting for top
|
|
Q3_TaskIDComplete( ent, TID_ANIM_LOWER );
|
|
}
|
|
else
|
|
{//Waiting for both to finish before complete
|
|
Q3_TaskIDClear( &ent->taskID[TID_ANIM_LOWER] );//Bottom is done, regardless
|
|
if ( !Q3_TaskIDPending( ent, TID_ANIM_UPPER) )
|
|
{//top is done and we're done
|
|
Q3_TaskIDComplete( ent, TID_ANIM_BOTH );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
-------------------------
|
|
PM_SetTorsoAnimTimer
|
|
-------------------------
|
|
*/
|
|
|
|
void PM_SetTorsoAnimTimer( gentity_t *ent, int *torsoAnimTimer, int time )
|
|
{
|
|
*torsoAnimTimer = time;
|
|
|
|
if ( *torsoAnimTimer < 0 && time != -1 )
|
|
{//Cap timer to 0 if was counting down, but let it be -1 if that was intentional
|
|
*torsoAnimTimer = 0;
|
|
}
|
|
|
|
if ( !*torsoAnimTimer && ent && Q3_TaskIDPending( ent, TID_ANIM_UPPER ) )
|
|
{//Waiting for torsoAnimTimer to complete, and it just got set to zero
|
|
if ( !Q3_TaskIDPending( ent, TID_ANIM_BOTH) )
|
|
{//Not waiting for bottom
|
|
Q3_TaskIDComplete( ent, TID_ANIM_UPPER );
|
|
}
|
|
else
|
|
{//Waiting for both to finish before complete
|
|
Q3_TaskIDClear( &ent->taskID[TID_ANIM_UPPER] );//Top is done, regardless
|
|
if ( !Q3_TaskIDPending( ent, TID_ANIM_LOWER) )
|
|
{//lower is done and we're done
|
|
Q3_TaskIDComplete( ent, TID_ANIM_BOTH );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
extern qboolean PM_SpinningSaberAnim( int anim );
|
|
extern float saberAnimSpeedMod[NUM_FORCE_POWER_LEVELS];
|
|
void PM_SaberStartTransAnim( int saberAnimLevel, int anim, float *animSpeed, gentity_t *gent )
|
|
{
|
|
if ( anim >= BOTH_A1_T__B_ && anim <= BOTH_ROLL_STAB )
|
|
{
|
|
if ( g_saberAnimSpeed->value != 1.0f )
|
|
{
|
|
*animSpeed *= g_saberAnimSpeed->value;
|
|
}
|
|
else if ( gent && gent->client && gent->client->ps.weapon == WP_SABER )
|
|
{
|
|
if ( gent->client->ps.saber[0].animSpeedScale != 1.0f )
|
|
{
|
|
*animSpeed *= gent->client->ps.saber[0].animSpeedScale;
|
|
}
|
|
if ( gent->client->ps.dualSabers
|
|
&& gent->client->ps.saber[1].animSpeedScale != 1.0f )
|
|
{
|
|
*animSpeed *= gent->client->ps.saber[1].animSpeedScale;
|
|
}
|
|
}
|
|
}
|
|
if ( gent
|
|
&& gent->client
|
|
&& gent->client->ps.stats[STAT_WEAPONS]&(1<<WP_SCEPTER)
|
|
&& gent->client->ps.dualSabers
|
|
&& saberAnimLevel == SS_DUAL
|
|
&& gent->weaponModel[1] )
|
|
{//using a scepter and dual style, slow down anims
|
|
if ( anim >= BOTH_A1_T__B_ && anim <= BOTH_H7_S7_BR )
|
|
{
|
|
*animSpeed *= 0.75;
|
|
}
|
|
}
|
|
if ( gent && gent->client && gent->client->ps.forceRageRecoveryTime > level.time )
|
|
{//rage recovery
|
|
if ( anim >= BOTH_A1_T__B_ && anim <= BOTH_H1_S1_BR )
|
|
{//animate slower
|
|
*animSpeed *= 0.75;
|
|
}
|
|
}
|
|
else if ( gent && gent->NPC && gent->NPC->rank == RANK_CIVILIAN )
|
|
{//grunt reborn
|
|
if ( anim >= BOTH_A1_T__B_ && anim <= BOTH_R1_TR_S1 )
|
|
{//his fast attacks are slower
|
|
if ( !PM_SpinningSaberAnim( anim ) )
|
|
{
|
|
*animSpeed *= 0.75;
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
else if ( gent && gent->client )
|
|
{
|
|
if ( gent->client->ps.saber[0].type == SABER_LANCE || gent->client->ps.saber[0].type == SABER_TRIDENT )
|
|
{//FIXME: hack for now - these use the fast anims, but slowed down. Should have own style
|
|
if ( anim >= BOTH_A1_T__B_ && anim <= BOTH_R1_TR_S1 )
|
|
{//his fast attacks are slower
|
|
if ( !PM_SpinningSaberAnim( anim ) )
|
|
{
|
|
*animSpeed *= 0.75;
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( ( anim >= BOTH_T1_BR__R &&
|
|
anim <= BOTH_T1_BL_TL ) ||
|
|
( anim >= BOTH_T3_BR__R &&
|
|
anim <= BOTH_T3_BL_TL ) ||
|
|
( anim >= BOTH_T5_BR__R &&
|
|
anim <= BOTH_T5_BL_TL ) )
|
|
{
|
|
if ( saberAnimLevel == FORCE_LEVEL_1 || saberAnimLevel == FORCE_LEVEL_5 )
|
|
{//FIXME: should not be necc for FORCE_LEVEL_1's
|
|
*animSpeed *= 1.5;
|
|
}
|
|
else if ( saberAnimLevel == FORCE_LEVEL_3 )
|
|
{
|
|
*animSpeed *= 0.75;
|
|
}
|
|
}
|
|
}
|
|
/*
|
|
void PM_SaberStartTransAnim( int anim, int entNum, int saberOffenseLevel, float *animSpeed )
|
|
{
|
|
//check starts
|
|
if ( ( anim >= BOTH_S1_S1_T_ &&
|
|
anim <= BOTH_S1_S1_TR ) ||
|
|
( anim >= BOTH_S1_S1_T_ &&
|
|
anim <= BOTH_S1_S1_TR ) ||
|
|
( anim >= BOTH_S3_S1_T_ &&
|
|
anim <= BOTH_S3_S1_TR ) )
|
|
{
|
|
if ( entNum == 0 )
|
|
{
|
|
*animSpeed *= saberAnimSpeedMod[FORCE_LEVEL_3];
|
|
}
|
|
else
|
|
{
|
|
*animSpeed *= saberAnimSpeedMod[saberOffenseLevel];
|
|
}
|
|
}
|
|
//Check transitions
|
|
else if ( PM_SpinningSaberAnim( anim ) )
|
|
{//spins stay normal speed
|
|
return;
|
|
}
|
|
else if ( ( anim >= BOTH_T1_BR__R &&
|
|
anim <= BOTH_T1_BL_TL ) ||
|
|
( anim >= BOTH_T2_BR__R &&
|
|
anim <= BOTH_T2_BL_TL ) ||
|
|
( anim >= BOTH_T3_BR__R &&
|
|
anim <= BOTH_T3_BL_TL ) )
|
|
{//slow down the transitions
|
|
if ( entNum == 0 && saberOffenseLevel <= FORCE_LEVEL_2 )
|
|
{
|
|
*animSpeed *= saberAnimSpeedMod[saberOffenseLevel];
|
|
}
|
|
else
|
|
{
|
|
*animSpeed *= saberAnimSpeedMod[saberOffenseLevel]/2.0f;
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
*/
|
|
extern qboolean player_locked;
|
|
extern qboolean MatrixMode;
|
|
float PM_GetTimeScaleMod( gentity_t *gent )
|
|
{
|
|
if ( g_timescale->value )
|
|
{
|
|
if ( !MatrixMode
|
|
&& gent->client->ps.legsAnim != BOTH_FORCELONGLEAP_START
|
|
&& gent->client->ps.legsAnim != BOTH_FORCELONGLEAP_ATTACK
|
|
&& gent->client->ps.legsAnim != BOTH_FORCELONGLEAP_LAND )
|
|
{
|
|
if ( gent && gent->s.clientNum == 0 && !player_locked && gent->client->ps.forcePowersActive&(1<<FP_SPEED) )
|
|
{
|
|
return (1.0 / g_timescale->value);
|
|
}
|
|
else if ( gent && gent->client && gent->client->ps.forcePowersActive&(1<<FP_SPEED) )
|
|
{
|
|
return (1.0 / g_timescale->value);
|
|
}
|
|
}
|
|
}
|
|
return 1.0f;
|
|
}
|
|
|
|
static inline qboolean PM_IsHumanoid( CGhoul2Info *ghlInfo )
|
|
{
|
|
char *GLAName;
|
|
GLAName = gi.G2API_GetGLAName( ghlInfo );
|
|
assert(GLAName);
|
|
|
|
if ( !Q_stricmp( "models/players/_humanoid/_humanoid", GLAName ) )
|
|
{
|
|
return qtrue;
|
|
}
|
|
|
|
return qfalse;
|
|
}
|
|
|
|
/*
|
|
-------------------------
|
|
PM_SetAnimFinal
|
|
-------------------------
|
|
*/
|
|
#define G2_DEBUG_TIMING (0)
|
|
void PM_SetAnimFinal(int *torsoAnim,int *legsAnim,
|
|
int setAnimParts,int anim,int setAnimFlags,
|
|
int *torsoAnimTimer,int *legsAnimTimer,
|
|
gentity_t *gent,int blendTime) // default blendTime=350
|
|
{
|
|
|
|
// BASIC SETUP AND SAFETY CHECKING
|
|
//=================================
|
|
|
|
// If It Is A Busted Entity, Don't Do Anything Here.
|
|
//---------------------------------------------------
|
|
if (!gent || !gent->client)
|
|
{
|
|
return;
|
|
}
|
|
|
|
// Make Sure This Character Has Such An Anim And A Model
|
|
//-------------------------------------------------------
|
|
if (anim<0 || anim>=MAX_ANIMATIONS || !ValidAnimFileIndex(gent->client->clientInfo.animFileIndex))
|
|
{
|
|
#ifndef FINAL_BUILD
|
|
if (g_AnimWarning->integer)
|
|
{
|
|
if (anim<0 || anim>=MAX_ANIMATIONS)
|
|
{
|
|
gi.Printf(S_COLOR_RED"PM_SetAnimFinal: Invalid Anim Index (%d)!\n", anim);
|
|
}
|
|
else
|
|
{
|
|
gi.Printf(S_COLOR_RED"PM_SetAnimFinal: Invalid Anim File Index (%d)!\n", gent->client->clientInfo.animFileIndex);
|
|
}
|
|
}
|
|
#endif
|
|
return;
|
|
}
|
|
|
|
|
|
// Get Global Time Properties
|
|
//----------------------------
|
|
float timeScaleMod = PM_GetTimeScaleMod( gent );
|
|
const int actualTime = (cg.time?cg.time:level.time);
|
|
const animation_t* animations = level.knownAnimFileSets[gent->client->clientInfo.animFileIndex].animations;
|
|
const animation_t& curAnim = animations[anim];
|
|
|
|
// Make Sure This Character Has Such An Anim And A Model
|
|
//-------------------------------------------------------
|
|
if (animations[anim].numFrames==0)
|
|
{
|
|
static int LastAnimWarningNum=0;
|
|
#ifndef FINAL_BUILD
|
|
if (LastAnimWarningNum!=anim)
|
|
{
|
|
if ((cg_debugAnim.integer==3) || // 3 = do everyone
|
|
(cg_debugAnim.integer==1 && gent->s.number==0) || // 1 = only the player
|
|
(cg_debugAnim.integer==2 && gent->s.number!=0) || // 2 = only everyone else
|
|
(cg_debugAnim.integer==4 && gent->s.number!=cg_debugAnimTarget.integer) // 4 = specific entnum
|
|
)
|
|
{
|
|
gi.Printf(S_COLOR_RED"PM_SetAnimFinal: Anim %s does not exist in this model (%s)!\n", animTable[anim].name, gent->NPC_type );
|
|
}
|
|
}
|
|
LastAnimWarningNum = anim;
|
|
#endif
|
|
return;
|
|
}
|
|
|
|
// If It's Not A Ghoul 2 Model, Just Remember The Anims And Stop, Because Everything Beyond This Is Ghoul2
|
|
//---------------------------------------------------------------------------------------------------------
|
|
if (!gi.G2API_HaveWeGhoul2Models(gent->ghoul2))
|
|
{
|
|
if (setAnimParts&SETANIM_TORSO)
|
|
{
|
|
(*torsoAnim) = anim;
|
|
}
|
|
if (setAnimParts&SETANIM_LEGS)
|
|
{
|
|
(*legsAnim) = anim;
|
|
}
|
|
return;
|
|
}
|
|
|
|
|
|
// Lower Offensive Skill Slows Down The Saber Start Attack Animations
|
|
//--------------------------------------------------------------------
|
|
PM_SaberStartTransAnim( gent->client->ps.saberAnimLevel, anim, &timeScaleMod, gent );
|
|
|
|
|
|
|
|
// SETUP VALUES FOR INCOMMING ANIMATION
|
|
//======================================
|
|
const bool animFootMove = (PM_WalkingAnim(anim) || PM_RunningAnim(anim) || anim==BOTH_CROUCH1WALK || anim==BOTH_CROUCH1WALKBACK);
|
|
const bool animHoldless = (setAnimFlags&SETANIM_FLAG_HOLDLESS)!=0;
|
|
const bool animHold = (setAnimFlags&SETANIM_FLAG_HOLD)!=0;
|
|
const bool animRestart = (setAnimFlags&SETANIM_FLAG_RESTART)!=0;
|
|
const bool animOverride = (setAnimFlags&SETANIM_FLAG_OVERRIDE)!=0;
|
|
const bool animSync = (g_synchSplitAnims->integer!=0 && !animRestart);
|
|
float animCurrent = (-1.0f);
|
|
float animSpeed = (50.0f / curAnim.frameLerp * timeScaleMod); // animSpeed is 1.0 if the frameLerp (ms/frame) is 50 (20 fps).
|
|
const float animFPS = (fabsf(curAnim.frameLerp));
|
|
const int animDurMSec = (int)(((curAnim.numFrames - 1) * animFPS) / timeScaleMod);
|
|
const int animHoldMSec = ((animHoldless && timeScaleMod==1.0f)?((animDurMSec>1)?(animDurMSec-1):(animFPS)):(animDurMSec));
|
|
int animFlags = (curAnim.loopFrames!=-1)?(BONE_ANIM_OVERRIDE_LOOP):(BONE_ANIM_OVERRIDE_FREEZE);
|
|
int animStart = (curAnim.firstFrame);
|
|
int animEnd = (curAnim.firstFrame)+(animations[anim].numFrames);
|
|
|
|
// If We Have A Blend Timer, Add The Blend Flag
|
|
//----------------------------------------------
|
|
if (blendTime > 0)
|
|
{
|
|
animFlags |= BONE_ANIM_BLEND;
|
|
}
|
|
|
|
// If Animation Is Going Backwards, Swap Last And First Frames
|
|
//-------------------------------------------------------------
|
|
if (animSpeed<0.0f)
|
|
{
|
|
// #ifndef FINAL_BUILD
|
|
#if 0
|
|
if (g_AnimWarning->integer==1)
|
|
{
|
|
if (animFlags&BONE_ANIM_OVERRIDE_LOOP)
|
|
{
|
|
gi.Printf(S_COLOR_YELLOW"PM_SetAnimFinal: WARNING: Anim (%s) looping backwards!\n", animTable[anim].name);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
int temp = animEnd;
|
|
animEnd = animStart;
|
|
animStart = temp;
|
|
blendTime = 0;
|
|
}
|
|
|
|
// If The Animation Is Walking Or Running, Attempt To Scale The Playback Speed To Match
|
|
//--------------------------------------------------------------------------------------
|
|
if (g_noFootSlide->integer
|
|
&& animFootMove
|
|
&& !(animSpeed<0.0f)
|
|
//FIXME: either read speed from animation.cfg or only do this for NPCs
|
|
// for whom we've specifically determined the proper numbers!
|
|
&& gent->client->NPC_class != CLASS_HOWLER
|
|
&& gent->client->NPC_class != CLASS_WAMPA
|
|
&& gent->client->NPC_class != CLASS_GONK
|
|
&& gent->client->NPC_class != CLASS_HOWLER
|
|
&& 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)
|
|
{
|
|
bool Walking = !!PM_WalkingAnim(anim);
|
|
bool HasDual = (gent->client->ps.saberAnimLevel==SS_DUAL);
|
|
bool HasStaff = (gent->client->ps.saberAnimLevel==SS_STAFF);
|
|
float moveSpeedOfAnim = 150.0f;//g_noFootSlideRunScale->value;
|
|
|
|
if (anim==BOTH_CROUCH1WALK || anim==BOTH_CROUCH1WALKBACK)
|
|
{
|
|
moveSpeedOfAnim = 75.0f;
|
|
}
|
|
else
|
|
{
|
|
if (gent->client->NPC_class == CLASS_HAZARD_TROOPER)
|
|
{
|
|
moveSpeedOfAnim = 50.0f;
|
|
}
|
|
else if (gent->client->NPC_class == CLASS_RANCOR)
|
|
{
|
|
moveSpeedOfAnim = 173.0f;
|
|
}
|
|
else
|
|
{
|
|
if (Walking)
|
|
{
|
|
if (HasDual || HasStaff)
|
|
{
|
|
moveSpeedOfAnim = 100.0f;
|
|
}
|
|
else
|
|
{
|
|
moveSpeedOfAnim = 50.0f;// g_noFootSlideWalkScale->value;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (HasStaff)
|
|
{
|
|
moveSpeedOfAnim = 250.0f;
|
|
}
|
|
else
|
|
{
|
|
moveSpeedOfAnim = 150.0f;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
animSpeed *= (gent->resultspeed/moveSpeedOfAnim);
|
|
if (animSpeed<0.01f)
|
|
{
|
|
animSpeed = 0.01f;
|
|
}
|
|
|
|
// Make Sure Not To Play Too Fast An Anim
|
|
//----------------------------------------
|
|
float maxPlaybackSpeed = (1.5f * timeScaleMod);
|
|
if (animSpeed>maxPlaybackSpeed)
|
|
{
|
|
animSpeed = maxPlaybackSpeed;
|
|
}
|
|
}
|
|
|
|
|
|
// GET VALUES FOR EXISTING BODY ANIMATION
|
|
//==========================================
|
|
float bodySpeed = 0.0f;
|
|
float bodyCurrent = 0.0f;
|
|
int bodyStart = 0;
|
|
int bodyEnd = 0;
|
|
int bodyFlags = 0;
|
|
int bodyAnim = (*legsAnim);
|
|
int bodyBone = (gent->rootBone);
|
|
bool bodyTimerOn = ((*legsAnimTimer>0) || (*legsAnimTimer)==-1);
|
|
bool bodyPlay = ((setAnimParts&SETANIM_LEGS) && (bodyBone!=-1) && (animOverride || !bodyTimerOn));
|
|
bool bodyAnimating = !!gi.G2API_GetBoneAnimIndex(&gent->ghoul2[gent->playerModel], bodyBone, actualTime, &bodyCurrent, &bodyStart, &bodyEnd, &bodyFlags, &bodySpeed, NULL);
|
|
bool bodyOnAnimNow = (bodyAnimating && bodyAnim==anim && bodyStart==animStart && bodyEnd==animEnd);
|
|
bool bodyMatchTorsFrame = false;
|
|
|
|
|
|
// GET VALUES FOR EXISTING TORSO ANIMATION
|
|
//===========================================
|
|
float torsSpeed = 0.0f;
|
|
float torsCurrent = 0.0f;
|
|
int torsStart = 0;
|
|
int torsEnd = 0;
|
|
int torsFlags = 0;
|
|
int torsAnim = (*torsoAnim);
|
|
int torsBone = (gent->lowerLumbarBone);
|
|
bool torsTimerOn = ((*torsoAnimTimer)>0 || (*torsoAnimTimer)==-1);
|
|
bool torsPlay = (gent->client->NPC_class!=CLASS_RANCOR && (setAnimParts&SETANIM_TORSO) && (torsBone!=-1) && (animOverride || !torsTimerOn));
|
|
bool torsAnimating = !!gi.G2API_GetBoneAnimIndex(&gent->ghoul2[gent->playerModel], torsBone, actualTime, &torsCurrent, &torsStart, &torsEnd, &torsFlags, &torsSpeed, NULL);
|
|
bool torsOnAnimNow = (torsAnimating && torsAnim==anim && torsStart==animStart && torsEnd==animEnd);
|
|
bool torsMatchBodyFrame = false;
|
|
|
|
|
|
// APPLY SYNC TO TORSO
|
|
//=====================
|
|
if (animSync && torsPlay && !bodyPlay && bodyOnAnimNow && (!torsOnAnimNow || torsCurrent!=bodyCurrent))
|
|
{
|
|
torsMatchBodyFrame = true;
|
|
animCurrent = bodyCurrent;
|
|
}
|
|
if (animSync && bodyPlay && !torsPlay && torsOnAnimNow && (!bodyOnAnimNow || bodyCurrent!=torsCurrent))
|
|
{
|
|
bodyMatchTorsFrame = true;
|
|
animCurrent = torsCurrent;
|
|
}
|
|
|
|
// If Already Doing These Exact Parameters, Then Don't Play
|
|
//----------------------------------------------------------
|
|
if (!animRestart)
|
|
{
|
|
torsPlay &= !(torsOnAnimNow && torsSpeed==animSpeed && !torsMatchBodyFrame);
|
|
bodyPlay &= !(bodyOnAnimNow && bodySpeed==animSpeed && !bodyMatchTorsFrame);
|
|
}
|
|
|
|
#ifndef FINAL_BUILD
|
|
if ((cg_debugAnim.integer==3) || // 3 = do everyone
|
|
(cg_debugAnim.integer==1 && gent->s.number==0) || // 1 = only the player
|
|
(cg_debugAnim.integer==2 && gent->s.number!=0) || // 2 = only everyone else
|
|
(cg_debugAnim.integer==4 && gent->s.number!=cg_debugAnimTarget.integer) // 4 = specific entnum
|
|
)
|
|
{
|
|
if (bodyPlay || torsPlay)
|
|
{
|
|
char* entName = gent->targetname;
|
|
char* location;
|
|
|
|
// Select Entity Name
|
|
//--------------------
|
|
if (!entName || !entName[0])
|
|
{
|
|
entName = gent->NPC_targetname;
|
|
}
|
|
if (!entName || !entName[0])
|
|
{
|
|
entName = gent->NPC_type;
|
|
}
|
|
if (!entName || !entName[0])
|
|
{
|
|
entName = gent->classname;
|
|
}
|
|
if (!entName || !entName[0])
|
|
{
|
|
entName = "UNKNOWN";
|
|
}
|
|
|
|
// Select Play Location
|
|
//----------------------
|
|
if (bodyPlay && torsPlay)
|
|
{
|
|
location = "BOTH ";
|
|
}
|
|
else if (bodyPlay)
|
|
{
|
|
location = "LEGS ";
|
|
}
|
|
else
|
|
{
|
|
location = "TORSO";
|
|
}
|
|
|
|
// Print It!
|
|
//-----------
|
|
Com_Printf("[%10d] ent[%3d-%18s] %s anim[%3d] - %s\n",
|
|
actualTime,
|
|
gent->s.number,
|
|
entName,
|
|
location,
|
|
anim,
|
|
animTable[anim].name );
|
|
}
|
|
}
|
|
#endif
|
|
|
|
|
|
// PLAY ON THE TORSO
|
|
//========================
|
|
if (torsPlay)
|
|
{
|
|
*torsoAnim = anim;
|
|
float oldAnimCurrent = animCurrent;
|
|
if (animCurrent!=bodyCurrent && torsOnAnimNow && !animRestart && !torsMatchBodyFrame)
|
|
{
|
|
animCurrent = torsCurrent;
|
|
}
|
|
|
|
gi.G2API_SetAnimIndex(&gent->ghoul2[gent->playerModel], curAnim.glaIndex);
|
|
gi.G2API_SetBoneAnimIndex(&gent->ghoul2[gent->playerModel], torsBone,
|
|
animStart,
|
|
animEnd,
|
|
(torsOnAnimNow && !animRestart)?(animFlags&~BONE_ANIM_BLEND):(animFlags),
|
|
animSpeed,
|
|
actualTime,
|
|
animCurrent,
|
|
blendTime);
|
|
|
|
if (gent->motionBone!=-1)
|
|
{
|
|
gi.G2API_SetBoneAnimIndex(&gent->ghoul2[gent->playerModel], gent->motionBone,
|
|
animStart,
|
|
animEnd,
|
|
(torsOnAnimNow && !animRestart)?(animFlags&~BONE_ANIM_BLEND):(animFlags),
|
|
animSpeed,
|
|
actualTime,
|
|
animCurrent,
|
|
blendTime);
|
|
}
|
|
|
|
animCurrent = oldAnimCurrent;
|
|
|
|
// If This Animation Is To Be Locked And Held, Calculate The Duration And Set The Timer
|
|
//--------------------------------------------------------------------------------------
|
|
if (animHold || animHoldless)
|
|
{
|
|
PM_SetTorsoAnimTimer(gent, torsoAnimTimer, animHoldMSec);
|
|
}
|
|
}
|
|
|
|
// PLAY ON THE WHOLE BODY
|
|
//========================
|
|
if (bodyPlay)
|
|
{
|
|
*legsAnim = anim;
|
|
|
|
if (bodyOnAnimNow && !animRestart && !bodyMatchTorsFrame)
|
|
{
|
|
animCurrent = bodyCurrent;
|
|
}
|
|
|
|
gi.G2API_SetAnimIndex(&gent->ghoul2[gent->playerModel], curAnim.glaIndex);
|
|
gi.G2API_SetBoneAnimIndex(&gent->ghoul2[gent->playerModel], bodyBone,
|
|
animStart,
|
|
animEnd,
|
|
(bodyOnAnimNow && !animRestart)?(animFlags&~BONE_ANIM_BLEND):(animFlags),
|
|
animSpeed,
|
|
actualTime,
|
|
animCurrent,
|
|
blendTime);
|
|
|
|
// If This Animation Is To Be Locked And Held, Calculate The Duration And Set The Timer
|
|
//--------------------------------------------------------------------------------------
|
|
if (animHold || animHoldless)
|
|
{
|
|
PM_SetLegsAnimTimer(gent, legsAnimTimer, animHoldMSec);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// PRINT SOME DEBUG TEXT OF EXISTING VALUES
|
|
//==========================================
|
|
if (false)
|
|
{
|
|
gi.Printf("PLAYANIM: (%3d) Speed(%4.2f) ", anim, animSpeed);
|
|
if (bodyAnimating)
|
|
{
|
|
gi.Printf("BODY: (%4.2f) (%4.2f) ", bodyCurrent, bodySpeed);
|
|
}
|
|
else
|
|
{
|
|
gi.Printf(" ");
|
|
}
|
|
if (torsAnimating)
|
|
{
|
|
gi.Printf("TORS: (%4.2f) (%4.2f)\n", torsCurrent, torsSpeed);
|
|
}
|
|
else
|
|
{
|
|
gi.Printf("\n");
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void PM_SetAnim(pmove_t *pm,int setAnimParts,int anim,int setAnimFlags, int blendTime)
|
|
{ // FIXME : once torsoAnim and legsAnim are in the same structure for NPC and Players
|
|
// rename PM_SetAnimFinal to PM_SetAnim and have both NPC and Players call PM_SetAnim
|
|
|
|
if ( pm->ps->pm_type >= PM_DEAD )
|
|
{//FIXME: sometimes we'll want to set anims when your dead... twitches, impacts, etc.
|
|
return;
|
|
}
|
|
|
|
if ( pm->gent == NULL )
|
|
{
|
|
return;
|
|
}
|
|
|
|
if ( !pm->gent || pm->gent->health > 0 )
|
|
{//don't lock anims if the guy is dead
|
|
if ( pm->ps->torsoAnimTimer
|
|
&& PM_LockedAnim( pm->ps->torsoAnim )
|
|
&& !PM_LockedAnim( anim ) )
|
|
{//nothing can override these special anims
|
|
setAnimParts &= ~SETANIM_TORSO;
|
|
}
|
|
|
|
if ( pm->ps->legsAnimTimer
|
|
&& PM_LockedAnim( pm->ps->legsAnim )
|
|
&& !PM_LockedAnim( anim ) )
|
|
{//nothing can override these special anims
|
|
setAnimParts &= ~SETANIM_LEGS;
|
|
}
|
|
}
|
|
|
|
if ( !setAnimParts )
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (setAnimFlags&SETANIM_FLAG_OVERRIDE)
|
|
{
|
|
// pm->ps->animationTimer = 0;
|
|
|
|
if (setAnimParts & SETANIM_TORSO)
|
|
{
|
|
if( (setAnimFlags & SETANIM_FLAG_RESTART) || pm->ps->torsoAnim != anim )
|
|
{
|
|
PM_SetTorsoAnimTimer( pm->gent, &pm->ps->torsoAnimTimer, 0 );
|
|
}
|
|
}
|
|
if (setAnimParts & SETANIM_LEGS)
|
|
{
|
|
if( (setAnimFlags & SETANIM_FLAG_RESTART) || pm->ps->legsAnim != anim )
|
|
{
|
|
PM_SetLegsAnimTimer( pm->gent, &pm->ps->legsAnimTimer, 0 );
|
|
}
|
|
}
|
|
}
|
|
|
|
PM_SetAnimFinal(&pm->ps->torsoAnim,&pm->ps->legsAnim,setAnimParts,anim,setAnimFlags,&pm->ps->torsoAnimTimer,&pm->ps->legsAnimTimer,&g_entities[pm->ps->clientNum],blendTime);//was pm->gent
|
|
}
|
|
|
|
bool TorsoAgainstWindTest( gentity_t* ent )
|
|
{
|
|
if (ent&&//valid ent
|
|
ent->client&&//a client
|
|
(ent->client->ps.weapon!=WP_SABER||ent->client->ps.saberMove==LS_READY)&&//either not holding a saber or the saber is in the ready pose
|
|
(ent->s.number<MAX_CLIENTS||G_ControlledByPlayer(ent)) &&
|
|
gi.WE_GetWindGusting(ent->currentOrigin) &&
|
|
gi.WE_IsOutside(ent->currentOrigin) )
|
|
{
|
|
if (Q_stricmp(level.mapname, "t2_wedge")!=0)
|
|
{
|
|
vec3_t fwd;
|
|
vec3_t windDir;
|
|
if (gi.WE_GetWindVector(windDir, ent->currentOrigin))
|
|
{
|
|
VectorScale(windDir, -1.0f, windDir);
|
|
AngleVectors(pm->gent->currentAngles, fwd, 0, 0);
|
|
if (DotProduct(fwd, windDir)>0.65f)
|
|
{
|
|
if (ent->client && ent->client->ps.torsoAnim!=BOTH_WIND)
|
|
{
|
|
NPC_SetAnim(ent, SETANIM_TORSO, BOTH_WIND, SETANIM_FLAG_NORMAL, 400);
|
|
}
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/*
|
|
-------------------------
|
|
PM_TorsoAnimLightsaber
|
|
-------------------------
|
|
*/
|
|
|
|
|
|
// Note that this function is intended to set the animation for the player, but
|
|
// only does idle-ish anims. Anything that has a timer associated, such as attacks and blocks,
|
|
// are set by PM_WeaponLightsaber()
|
|
|
|
extern Vehicle_t *G_IsRidingVehicle( gentity_t *pEnt );
|
|
extern qboolean PM_LandingAnim( int anim );
|
|
extern qboolean PM_JumpingAnim( int anim );
|
|
qboolean PM_InCartwheel( int anim );
|
|
void PM_TorsoAnimLightsaber()
|
|
{
|
|
// *********************************************************
|
|
// WEAPON_READY
|
|
// *********************************************************
|
|
if ( pm->ps->forcePowersActive&(1<<FP_GRIP) && pm->ps->forcePowerLevel[FP_GRIP] > FORCE_LEVEL_1 )
|
|
{//holding an enemy aloft with force-grip
|
|
return;
|
|
}
|
|
|
|
if ( pm->ps->forcePowersActive&(1<<FP_LIGHTNING) && pm->ps->forcePowerLevel[FP_LIGHTNING] > FORCE_LEVEL_1 )
|
|
{//lightning
|
|
return;
|
|
}
|
|
|
|
if ( pm->ps->forcePowersActive&(1<<FP_DRAIN) )
|
|
{//drain
|
|
return;
|
|
}
|
|
|
|
if ( pm->ps->saber[0].blade[0].active
|
|
&& pm->ps->saber[0].blade[0].length < 3
|
|
&& !(pm->ps->saberEventFlags&SEF_HITWALL)
|
|
&& pm->ps->weaponstate == WEAPON_RAISING )
|
|
{
|
|
if (!G_IsRidingVehicle(pm->gent))
|
|
{
|
|
PM_SetSaberMove(LS_DRAW);
|
|
}
|
|
return;
|
|
}
|
|
else if ( !pm->ps->SaberActive() && pm->ps->SaberLength() )
|
|
{
|
|
if (!G_IsRidingVehicle(pm->gent))
|
|
{
|
|
PM_SetSaberMove(LS_PUTAWAY);
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (pm->ps->weaponTime > 0)
|
|
{ // weapon is already busy.
|
|
if ( pm->ps->torsoAnim == BOTH_TOSS1
|
|
|| pm->ps->torsoAnim == BOTH_TOSS2 )
|
|
{//in toss
|
|
if ( !pm->ps->torsoAnimTimer )
|
|
{//weird, get out of it, I guess
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
|
|
if ( pm->ps->weaponstate == WEAPON_READY ||
|
|
pm->ps->weaponstate == WEAPON_CHARGING ||
|
|
pm->ps->weaponstate == WEAPON_CHARGING_ALT )
|
|
{//ready
|
|
if ( pm->ps->weapon == WP_SABER && (pm->ps->SaberLength()) )
|
|
{//saber is on
|
|
// Select the proper idle Lightsaber attack move from the chart.
|
|
if (pm->ps->saberMove > LS_READY && pm->ps->saberMove < LS_MOVE_MAX)
|
|
{
|
|
PM_SetSaberMove(saberMoveData[pm->ps->saberMove].chain_idle);
|
|
}
|
|
else
|
|
{
|
|
if ( PM_JumpingAnim( pm->ps->legsAnim )
|
|
|| PM_LandingAnim( pm->ps->legsAnim )
|
|
|| PM_InCartwheel( pm->ps->legsAnim )
|
|
|| PM_FlippingAnim( pm->ps->legsAnim ))
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && pm->ps->torsoAnim == BOTH_BUTTON_HOLD )
|
|
{//using something
|
|
if ( !pm->ps->useTime )
|
|
{//stopped holding it, release
|
|
PM_SetAnim( pm, SETANIM_TORSO, BOTH_BUTTON_RELEASE, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD );
|
|
}//else still holding, leave it as it is
|
|
}
|
|
else
|
|
{
|
|
if ( (PM_RunningAnim( pm->ps->legsAnim )
|
|
|| pm->ps->legsAnim == BOTH_WALK_STAFF
|
|
|| pm->ps->legsAnim == BOTH_WALK_DUAL
|
|
|| pm->ps->legsAnim == BOTH_WALKBACK_STAFF
|
|
|| pm->ps->legsAnim == BOTH_WALKBACK_DUAL )
|
|
&& pm->ps->saberBlockingTime < cg.time )
|
|
{//running w/1-handed weapon uses full-body anim
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
PM_SetSaberMove(LS_READY);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
/*
|
|
if ( PM_JumpingAnim( pm->ps->legsAnim )
|
|
|| PM_LandingAnim( pm->ps->legsAnim )
|
|
|| PM_InCartwheel( pm->ps->legsAnim )
|
|
|| PM_FlippingAnim( pm->ps->legsAnim ))
|
|
{//jumping, landing cartwheel, flipping
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
PM_SetSaberMove( LS_READY );
|
|
}
|
|
*/
|
|
}
|
|
else if (TorsoAgainstWindTest(pm->gent))
|
|
{
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_RUN1 )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_RUN1,SETANIM_FLAG_NORMAL);
|
|
pm->ps->saberMove = LS_READY;
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_RUN2 )//&& pm->ps->saberAnimLevel != SS_STAFF )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_RUN2,SETANIM_FLAG_NORMAL);
|
|
pm->ps->saberMove = LS_READY;
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_RUN_STAFF )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_RUN_STAFF,SETANIM_FLAG_NORMAL);
|
|
pm->ps->saberMove = LS_READY;
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_RUN_DUAL )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_RUN_DUAL,SETANIM_FLAG_NORMAL);
|
|
pm->ps->saberMove = LS_READY;
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_WALK1 )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_WALK1,SETANIM_FLAG_NORMAL);
|
|
pm->ps->saberMove = LS_READY;
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_WALK2 )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_WALK2,SETANIM_FLAG_NORMAL);
|
|
pm->ps->saberMove = LS_READY;
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_WALK_STAFF )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_WALK_STAFF,SETANIM_FLAG_NORMAL);
|
|
pm->ps->saberMove = LS_READY;
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_WALK_DUAL )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_WALK_DUAL,SETANIM_FLAG_NORMAL);
|
|
pm->ps->saberMove = LS_READY;
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_CROUCH1IDLE && pm->ps->clientNum != 0 )//player falls through
|
|
{
|
|
//??? Why nothing? What if you were running???
|
|
//PM_SetAnim(pm,SETANIM_TORSO,BOTH_CROUCH1IDLE,SETANIM_FLAG_NORMAL);
|
|
pm->ps->saberMove = LS_READY;
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_JUMP1 )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_JUMP1,SETANIM_FLAG_NORMAL);
|
|
pm->ps->saberMove = LS_READY;
|
|
}
|
|
else
|
|
{//Used to default to both_stand1 which is an arms-down anim
|
|
// PM_SetAnim(pm,SETANIM_TORSO,BOTH_ATTACK1,SETANIM_FLAG_NORMAL);//TORSO_WEAPONREADY1
|
|
// Select the next proper pose for the lightsaber assuming that there are no attacks.
|
|
if (pm->ps->saberMove > LS_READY && pm->ps->saberMove < LS_MOVE_MAX)
|
|
{
|
|
PM_SetSaberMove(saberMoveData[pm->ps->saberMove].chain_idle);
|
|
}
|
|
else
|
|
{
|
|
if ( PM_JumpingAnim( pm->ps->legsAnim )
|
|
|| PM_LandingAnim( pm->ps->legsAnim )
|
|
|| PM_InCartwheel( pm->ps->legsAnim )
|
|
|| PM_FlippingAnim( pm->ps->legsAnim ))
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && pm->ps->torsoAnim == BOTH_BUTTON_HOLD )
|
|
{//using something
|
|
if ( !pm->ps->useTime )
|
|
{//stopped holding it, release
|
|
PM_SetAnim( pm, SETANIM_TORSO, BOTH_BUTTON_RELEASE, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD );
|
|
}//else still holding, leave it as it is
|
|
}
|
|
else
|
|
{
|
|
PM_SetSaberMove(LS_READY);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// *********************************************************
|
|
// WEAPON_IDLE
|
|
// *********************************************************
|
|
|
|
else if ( pm->ps->weaponstate == WEAPON_IDLE )
|
|
{
|
|
if (TorsoAgainstWindTest(pm->gent))
|
|
{
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_GUARD_LOOKAROUND1 )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_GUARD_LOOKAROUND1,SETANIM_FLAG_NORMAL);
|
|
pm->ps->saberMove = LS_READY;
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_GUARD_IDLE1 )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_GUARD_IDLE1,SETANIM_FLAG_NORMAL);
|
|
pm->ps->saberMove = LS_READY;
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_STAND1IDLE1
|
|
|| pm->ps->legsAnim == BOTH_STAND2IDLE1
|
|
|| pm->ps->legsAnim == BOTH_STAND2IDLE2
|
|
|| pm->ps->legsAnim == BOTH_STAND3IDLE1
|
|
|| pm->ps->legsAnim == BOTH_STAND5IDLE1 )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
pm->ps->saberMove = LS_READY;
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_STAND2TO4 )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_STAND2TO4,SETANIM_FLAG_NORMAL);
|
|
pm->ps->saberMove = LS_READY;
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_STAND4TO2 )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_STAND4TO2,SETANIM_FLAG_NORMAL);
|
|
pm->ps->saberMove = LS_READY;
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_STAND4 )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_STAND4,SETANIM_FLAG_NORMAL);
|
|
pm->ps->saberMove = LS_READY;
|
|
}
|
|
else
|
|
{
|
|
// This is now set in SetSaberMove.
|
|
// Idle for Lightsaber
|
|
if ( pm->gent && pm->gent->client )
|
|
{
|
|
// pm->gent->client->saberTrail.inAction = qfalse;
|
|
}
|
|
|
|
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->saberInFlight
|
|
&& saberInAir
|
|
&& (!pm->ps->dualSabers || !pm->ps->saber[1].Active()))
|
|
{
|
|
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
|
|
{//saber is on
|
|
// Idle for Lightsaber
|
|
if ( pm->gent && pm->gent->client )
|
|
{
|
|
if ( !G_InCinematicSaberAnim( pm->gent ) )
|
|
{
|
|
pm->gent->client->ps.SaberDeactivateTrail( 0 );
|
|
}
|
|
}
|
|
// Idle for idle/ready Lightsaber
|
|
// PM_SetAnim(pm,SETANIM_TORSO,BOTH_ATTACK1,SETANIM_FLAG_NORMAL);//TORSO_WEAPONIDLE1
|
|
// Select the proper idle Lightsaber attack move from the chart.
|
|
if (pm->ps->saberMove > LS_READY && pm->ps->saberMove < LS_MOVE_MAX)
|
|
{
|
|
PM_SetSaberMove(saberMoveData[pm->ps->saberMove].chain_idle);
|
|
}
|
|
else
|
|
{
|
|
if ( PM_JumpingAnim( pm->ps->legsAnim )
|
|
|| PM_LandingAnim( pm->ps->legsAnim )
|
|
|| PM_InCartwheel( pm->ps->legsAnim )
|
|
|| PM_FlippingAnim( pm->ps->legsAnim ))
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && pm->ps->torsoAnim == BOTH_BUTTON_HOLD )
|
|
{//using something
|
|
if ( !pm->ps->useTime )
|
|
{//stopped holding it, release
|
|
PM_SetAnim( pm, SETANIM_TORSO, BOTH_BUTTON_RELEASE, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD );
|
|
}//else still holding, leave it as it is
|
|
}
|
|
else
|
|
{
|
|
if ( (PM_RunningAnim( pm->ps->legsAnim )
|
|
|| pm->ps->legsAnim == BOTH_WALK_STAFF
|
|
|| pm->ps->legsAnim == BOTH_WALK_DUAL
|
|
|| pm->ps->legsAnim == BOTH_WALKBACK_STAFF
|
|
|| pm->ps->legsAnim == BOTH_WALKBACK_DUAL )
|
|
&& pm->ps->saberBlockingTime < cg.time )
|
|
{//running w/1-handed weapon uses full-body anim
|
|
int setFlags = SETANIM_FLAG_NORMAL;
|
|
if ( PM_LandingAnim( pm->ps->torsoAnim ) )
|
|
{
|
|
setFlags = SETANIM_FLAG_OVERRIDE;
|
|
}
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,setFlags);
|
|
}
|
|
else
|
|
{
|
|
PM_SetSaberMove(LS_READY);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
-------------------------
|
|
PM_TorsoAnimation
|
|
-------------------------
|
|
*/
|
|
|
|
void PM_TorsoAnimation( void )
|
|
{//FIXME: Write a much smarter and more appropriate anim picking routine logic...
|
|
// int oldAnim;
|
|
if ( PM_InKnockDown( pm->ps ) || PM_InRoll( pm->ps ))
|
|
{//in knockdown
|
|
return;
|
|
}
|
|
|
|
if ( (pm->ps->eFlags&EF_HELD_BY_WAMPA) )
|
|
{
|
|
return;
|
|
}
|
|
|
|
if ( (pm->ps->eFlags&EF_FORCE_DRAINED) )
|
|
{//being drained
|
|
//PM_SetAnim( pm, SETANIM_TORSO, BOTH_HUGGEE1, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD );
|
|
return;
|
|
}
|
|
if ( (pm->ps->forcePowersActive&(1<<FP_DRAIN))
|
|
&& pm->ps->forceDrainEntityNum < ENTITYNUM_WORLD )
|
|
{//draining
|
|
//PM_SetAnim( pm, SETANIM_TORSO, BOTH_HUGGER1, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD );
|
|
return;
|
|
}
|
|
|
|
if( pm->gent && pm->gent->NPC && (pm->gent->NPC->scriptFlags & SCF_FORCED_MARCH) )
|
|
{
|
|
return;
|
|
}
|
|
|
|
if(pm->gent != NULL && pm->gent->client)
|
|
{
|
|
pm->gent->client->renderInfo.torsoFpsMod = 1.0f;
|
|
}
|
|
|
|
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
|
|
{//torso
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_GUNSIT1,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD);//SETANIM_FLAG_NORMAL
|
|
}
|
|
return;
|
|
}
|
|
/* else if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_VEHICLE && pm->ps->clientNum < MAX_CLIENTS && (m_pVehicleInfo[((CVehicleNPC *)pm->gent->NPC)->m_iVehicleTypeID].numHands == 2 || g_speederControlScheme->value == 2) )
|
|
{//can't look around
|
|
PM_SetAnim(pm,SETANIM_TORSO,m_pVehicleInfo[((CVehicleNPC *)pm->gent->NPC)->m_iVehicleTypeID].riderAnim,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD);
|
|
return;
|
|
}*/
|
|
|
|
if ( pm->ps->taunting > level.time )
|
|
{
|
|
if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_ALORA )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_BOTH,BOTH_ALORA_TAUNT,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD);//SETANIM_FLAG_NORMAL
|
|
}
|
|
else if ( pm->ps->weapon == WP_SABER && pm->ps->saberAnimLevel == SS_DUAL && PM_HasAnimation( pm->gent, BOTH_DUAL_TAUNT ) )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_BOTH,BOTH_DUAL_TAUNT,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD);//SETANIM_FLAG_NORMAL
|
|
}
|
|
else if ( pm->ps->weapon == WP_SABER
|
|
&& pm->ps->saberAnimLevel == SS_STAFF )//pm->ps->saber[0].type == SABER_STAFF )
|
|
{//turn on the blades
|
|
if ( PM_HasAnimation( pm->gent, BOTH_STAFF_TAUNT ) )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_BOTH,BOTH_STAFF_TAUNT,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD);//SETANIM_FLAG_NORMAL
|
|
}
|
|
/*
|
|
else
|
|
{
|
|
if ( !pm->ps->saber[0].blade[0].active )
|
|
{//first blade is off
|
|
//turn it on
|
|
pm->ps->SaberBladeActivate( 0, 0, qtrue );
|
|
if ( !pm->ps->saber[0].blade[1].active )
|
|
{//second blade is also off, extend time of this taunt so we have enough time to turn them both on
|
|
pm->ps->taunting = level.time + 3000;
|
|
}
|
|
}
|
|
else if ( (pm->ps->taunting - level.time) < 1500 )
|
|
{//only 1500ms left in taunt
|
|
if ( !pm->ps->saber[0].blade[1].active )
|
|
{//second blade is off
|
|
//turn it on
|
|
pm->ps->SaberBladeActivate( 0, 1, qtrue );
|
|
}
|
|
}
|
|
//pose
|
|
PM_SetAnim(pm,SETANIM_BOTH,BOTH_SABERSTAFF_STANCE,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD);
|
|
pm->ps->torsoAnimTimer = pm->ps->legsAnimTimer = (pm->ps->taunting - level.time);
|
|
}
|
|
*/
|
|
}
|
|
else if ( PM_HasAnimation( pm->gent, BOTH_GESTURE1 ) )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_BOTH,BOTH_GESTURE1,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD);//SETANIM_FLAG_NORMAL
|
|
pm->gent->client->ps.SaberActivateTrail( 100 );
|
|
//FIXME: will this reset?
|
|
//FIXME: force-control (yellow glow) effect on hand and saber?
|
|
}
|
|
else
|
|
{
|
|
//PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONIDLE1,SETANIM_FLAG_NORMAL);
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (pm->ps->weapon == WP_SABER ) // WP_LIGHTSABER
|
|
{
|
|
qboolean saberInAir = qfalse;
|
|
if ( pm->ps->SaberLength() && !pm->ps->saberInFlight )
|
|
{
|
|
PM_TorsoAnimLightsaber();
|
|
}
|
|
else
|
|
{
|
|
if ( pm->ps->forcePowersActive&(1<<FP_GRIP) && pm->ps->forcePowerLevel[FP_GRIP] > FORCE_LEVEL_1 )
|
|
{//holding an enemy aloft with force-grip
|
|
return;
|
|
}
|
|
if ( pm->ps->forcePowersActive&(1<<FP_LIGHTNING) && pm->ps->forcePowerLevel[FP_LIGHTNING] > FORCE_LEVEL_1 )
|
|
{//lightning
|
|
return;
|
|
}
|
|
if ( pm->ps->forcePowersActive&(1<<FP_DRAIN) )
|
|
{//drain
|
|
return;
|
|
}
|
|
|
|
saberInAir = qtrue;
|
|
|
|
if ( PM_SaberInBrokenParry( pm->ps->saberMove ) || pm->ps->saberBlocked == BLOCKED_PARRY_BROKEN || PM_DodgeAnim( pm->ps->torsoAnim ) )
|
|
{//we're stuck in a broken parry
|
|
PM_TorsoAnimLightsaber();
|
|
}
|
|
else
|
|
{
|
|
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->saberInFlight
|
|
&& saberInAir
|
|
&& (!pm->ps->dualSabers //not using 2 sabers
|
|
|| !pm->ps->saber[1].Active() //left one off
|
|
|| pm->ps->torsoAnim == BOTH_SABERDUAL_STANCE//not attacking
|
|
|| pm->ps->torsoAnim == BOTH_SABERPULL//not attacking
|
|
|| pm->ps->torsoAnim == BOTH_STAND1//not attacking
|
|
|| PM_RunningAnim( pm->ps->torsoAnim ) //not attacking
|
|
|| PM_WalkingAnim( pm->ps->torsoAnim ) //not attacking
|
|
|| PM_JumpingAnim( pm->ps->torsoAnim )//not attacking
|
|
|| PM_SwimmingAnim( pm->ps->torsoAnim ) )//not attacking
|
|
)
|
|
{
|
|
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_InSlopeAnim( pm->ps->legsAnim ) )
|
|
{//HMM... this probably breaks the saber putaway and select anims
|
|
if ( pm->ps->SaberLength() > 0 )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_STAND2,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_STAND1,SETANIM_FLAG_NORMAL);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && pm->ps->torsoAnim == BOTH_BUTTON_HOLD )
|
|
{//using something
|
|
if ( !pm->ps->useTime )
|
|
{//stopped holding it, release
|
|
PM_SetAnim( pm, SETANIM_TORSO, BOTH_BUTTON_RELEASE, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD );
|
|
}//else still holding, leave it as it is
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (pm->ps->weaponTime<= 0 && (pm->ps->saberMove==LS_READY || pm->ps->SaberLength()==0) && !saberInAir)
|
|
{
|
|
TorsoAgainstWindTest(pm->gent);
|
|
}
|
|
return;
|
|
}
|
|
|
|
if ( PM_ForceAnim( pm->ps->torsoAnim )
|
|
&& pm->ps->torsoAnimTimer > 0 )
|
|
{//in a force anim, don't do a stand anim
|
|
return;
|
|
}
|
|
|
|
|
|
qboolean weaponBusy = qfalse;
|
|
|
|
if ( pm->ps->weapon == WP_NONE )
|
|
{
|
|
weaponBusy = qfalse;
|
|
}
|
|
else if ( pm->ps->weaponstate == WEAPON_FIRING || pm->ps->weaponstate == WEAPON_CHARGING || pm->ps->weaponstate == WEAPON_CHARGING_ALT )
|
|
{
|
|
weaponBusy = qtrue;
|
|
}
|
|
else if ( pm->ps->lastShotTime > level.time - 3000 )
|
|
{
|
|
weaponBusy = qtrue;
|
|
}
|
|
else if ( pm->ps->weaponTime > 0 )
|
|
{
|
|
weaponBusy = qtrue;
|
|
}
|
|
else if ( pm->gent && pm->gent->client->fireDelay > 0 )
|
|
{
|
|
weaponBusy = qtrue;
|
|
}
|
|
else if ( TorsoAgainstWindTest(pm->gent) )
|
|
{
|
|
return;
|
|
}
|
|
else if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && cg.zoomTime > cg.time - 5000 )
|
|
{//if we used binoculars recently, aim weapon
|
|
weaponBusy = qtrue;
|
|
pm->ps->weaponstate = WEAPON_IDLE;
|
|
}
|
|
else if ( pm->ps->pm_flags & PMF_DUCKED )
|
|
{//ducking is considered on alert... plus looks stupid to have arms hanging down when crouched
|
|
weaponBusy = qtrue;
|
|
}
|
|
|
|
if ( pm->ps->weapon == WP_NONE ||
|
|
pm->ps->weaponstate == WEAPON_READY ||
|
|
pm->ps->weaponstate == WEAPON_CHARGING ||
|
|
pm->ps->weaponstate == WEAPON_CHARGING_ALT )
|
|
{
|
|
if ( pm->ps->weapon == WP_SABER && pm->ps->SaberLength() )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_ATTACK1,SETANIM_FLAG_NORMAL);//TORSO_WEAPONREADY1
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_RUN1 && !weaponBusy )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_RUN1,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_RUN2 && !weaponBusy )//&& pm->ps->saberAnimLevel != SS_STAFF )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_RUN2,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_RUN4 && !weaponBusy )//&& pm->ps->saberAnimLevel != SS_STAFF )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_RUN4,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_RUN_STAFF && !weaponBusy )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_RUN_STAFF,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_RUN_DUAL && !weaponBusy )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_RUN_DUAL,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_WALK1 && !weaponBusy )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_WALK1,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_WALK2 && !weaponBusy )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_WALK2,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_WALK_STAFF && !weaponBusy )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_WALK_STAFF,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_WALK_DUAL&& !weaponBusy )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_WALK_DUAL,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_CROUCH1IDLE && pm->ps->clientNum != 0 )//player falls through
|
|
{
|
|
//??? Why nothing? What if you were running???
|
|
//PM_SetAnim(pm,SETANIM_TORSO,BOTH_CROUCH1IDLE,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_JUMP1 && !weaponBusy )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_JUMP1,SETANIM_FLAG_NORMAL, 100); // Only blend over 100ms
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_SWIM_IDLE1 && !weaponBusy )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_SWIM_IDLE1,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_SWIMFORWARD && !weaponBusy )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_SWIMFORWARD,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if ( pm->ps->weapon == WP_NONE )
|
|
{
|
|
int legsAnim = pm->ps->legsAnim;
|
|
/*
|
|
if ( PM_RollingAnim( legsAnim ) ||
|
|
PM_FlippingAnim( legsAnim ) ||
|
|
PM_JumpingAnim( legsAnim ) ||
|
|
PM_PainAnim( legsAnim ) ||
|
|
PM_SwimmingAnim( legsAnim ) )
|
|
*/
|
|
{
|
|
PM_SetAnim(pm, SETANIM_TORSO, legsAnim, SETANIM_FLAG_NORMAL );
|
|
}
|
|
}
|
|
else
|
|
{//Used to default to both_stand1 which is an arms-down anim
|
|
if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && pm->ps->torsoAnim == BOTH_BUTTON_HOLD )
|
|
{//using something
|
|
if ( !pm->ps->useTime )
|
|
{//stopped holding it, release
|
|
PM_SetAnim( pm, SETANIM_TORSO, BOTH_BUTTON_RELEASE, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD );
|
|
}//else still holding, leave it as it is
|
|
}
|
|
else if ( pm->gent != NULL
|
|
&& (pm->gent->s.number<MAX_CLIENTS||G_ControlledByPlayer(pm->gent))
|
|
&& pm->ps->weaponstate != WEAPON_CHARGING
|
|
&& pm->ps->weaponstate != WEAPON_CHARGING_ALT )
|
|
{//PLayer- temp hack for weapon frame
|
|
if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_RANCOR )
|
|
{//ignore
|
|
}
|
|
else if ( pm->ps->weapon == WP_MELEE )
|
|
{//hehe
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_STAND6,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_STAND1,SETANIM_FLAG_NORMAL);
|
|
}
|
|
}
|
|
else if ( PM_InSpecialJump( pm->ps->legsAnim ) )
|
|
{//use legs anim
|
|
//FIXME: or just use whatever's currently playing?
|
|
//PM_SetAnim( pm, SETANIM_TORSO, pm->ps->legsAnim, SETANIM_FLAG_NORMAL );
|
|
}
|
|
else
|
|
{
|
|
switch(pm->ps->weapon)
|
|
{
|
|
// ********************************************************
|
|
case WP_SABER: // WP_LIGHTSABER
|
|
// Ready pose for Lightsaber
|
|
// PM_SetAnim(pm,SETANIM_TORSO,BOTH_ATTACK1,SETANIM_FLAG_NORMAL);//TORSO_WEAPONREADY1
|
|
// Select the next proper pose for the lightsaber assuming that there are no attacks.
|
|
if (pm->ps->saberMove > LS_NONE && pm->ps->saberMove < LS_MOVE_MAX)
|
|
{
|
|
PM_SetSaberMove(saberMoveData[pm->ps->saberMove].chain_idle);
|
|
}
|
|
break;
|
|
// ********************************************************
|
|
|
|
case WP_BRYAR_PISTOL:
|
|
//FIXME: if recently fired, hold the ready!
|
|
if ( pm->ps->weaponstate == WEAPON_CHARGING_ALT || weaponBusy )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONREADY2,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if ( PM_RunningAnim( pm->ps->legsAnim )
|
|
|| PM_WalkingAnim( pm->ps->legsAnim )
|
|
|| PM_JumpingAnim( pm->ps->legsAnim )
|
|
|| PM_SwimmingAnim( pm->ps->legsAnim ) )
|
|
{//running w/1-handed weapon uses full-body anim
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONREADY2,SETANIM_FLAG_NORMAL);
|
|
}
|
|
break;
|
|
case WP_BLASTER_PISTOL:
|
|
if ( pm->gent
|
|
&& pm->gent->weaponModel[1] > 0 )
|
|
{//dual pistols
|
|
if ( weaponBusy )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_GUNSIT1,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if ( PM_RunningAnim( pm->ps->legsAnim )
|
|
|| PM_WalkingAnim( pm->ps->legsAnim )
|
|
|| PM_JumpingAnim( pm->ps->legsAnim )
|
|
|| PM_SwimmingAnim( pm->ps->legsAnim ) )
|
|
{//running w/1-handed weapon uses full-body anim
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_STAND6,SETANIM_FLAG_NORMAL);
|
|
}
|
|
}
|
|
else
|
|
{//single pistols
|
|
if ( pm->ps->weaponstate == WEAPON_CHARGING_ALT || weaponBusy )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONREADY2,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if ( PM_RunningAnim( pm->ps->legsAnim )
|
|
|| PM_WalkingAnim( pm->ps->legsAnim )
|
|
|| PM_JumpingAnim( pm->ps->legsAnim )
|
|
|| PM_SwimmingAnim( pm->ps->legsAnim ) )
|
|
{//running w/1-handed weapon uses full-body anim
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONREADY2,SETANIM_FLAG_NORMAL);
|
|
}
|
|
}
|
|
break;
|
|
case WP_NONE:
|
|
//NOTE: should never get here
|
|
break;
|
|
case WP_MELEE:
|
|
if ( PM_RunningAnim( pm->ps->legsAnim )
|
|
|| PM_WalkingAnim( pm->ps->legsAnim )
|
|
|| PM_JumpingAnim( pm->ps->legsAnim )
|
|
|| PM_SwimmingAnim( pm->ps->legsAnim ) )
|
|
{//running w/1-handed weapon uses full-body anim
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_RANCOR )
|
|
{//ignore
|
|
}
|
|
else if ( pm->gent && pm->gent->client && !PM_DroidMelee( pm->gent->client->NPC_class ) )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_STAND6,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_STAND1,SETANIM_FLAG_NORMAL);
|
|
}
|
|
}
|
|
break;
|
|
case WP_TUSKEN_STAFF:
|
|
if ( PM_RunningAnim( pm->ps->legsAnim )
|
|
|| PM_WalkingAnim( pm->ps->legsAnim )
|
|
|| PM_JumpingAnim( pm->ps->legsAnim )
|
|
|| PM_SwimmingAnim( pm->ps->legsAnim ) )
|
|
{//running w/1-handed weapon uses full-body anim
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim(pm, SETANIM_TORSO, BOTH_STAND3, SETANIM_FLAG_NORMAL);
|
|
}
|
|
break;
|
|
|
|
case WP_NOGHRI_STICK:
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONREADY3,SETANIM_FLAG_NORMAL);
|
|
//PM_SetAnim(pm,SETANIM_LEGS,BOTH_ATTACK2,SETANIM_FLAG_NORMAL);
|
|
break;
|
|
|
|
case WP_BLASTER:
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONREADY3,SETANIM_FLAG_NORMAL);
|
|
//PM_SetAnim(pm,SETANIM_LEGS,BOTH_ATTACK2,SETANIM_FLAG_NORMAL);
|
|
break;
|
|
case WP_DISRUPTOR:
|
|
case WP_TUSKEN_RIFLE:
|
|
if ( (pm->ps->weaponstate != WEAPON_FIRING
|
|
&& pm->ps->weaponstate != WEAPON_CHARGING
|
|
&& pm->ps->weaponstate != WEAPON_CHARGING_ALT)
|
|
|| PM_RunningAnim( pm->ps->legsAnim )
|
|
|| PM_WalkingAnim( pm->ps->legsAnim )
|
|
|| PM_JumpingAnim( pm->ps->legsAnim )
|
|
|| PM_SwimmingAnim( pm->ps->legsAnim ) )
|
|
{//running sniper weapon uses normal ready
|
|
if ( pm->ps->clientNum )
|
|
{
|
|
PM_SetAnim( pm, SETANIM_TORSO, TORSO_WEAPONREADY3, SETANIM_FLAG_NORMAL );
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim( pm, SETANIM_TORSO, TORSO_WEAPONREADY3, SETANIM_FLAG_NORMAL );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ( pm->ps->clientNum )
|
|
{
|
|
PM_SetAnim( pm, SETANIM_TORSO, TORSO_WEAPONREADY4, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD );//TORSO_WEAPONREADY4//SETANIM_FLAG_RESTART|
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim( pm, SETANIM_TORSO, TORSO_WEAPONREADY4, SETANIM_FLAG_NORMAL );
|
|
}
|
|
}
|
|
break;
|
|
case WP_BOT_LASER:
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONIDLE2,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_RESTART|SETANIM_FLAG_HOLD);
|
|
break;
|
|
case WP_THERMAL:
|
|
if ( pm->ps->weaponstate != WEAPON_FIRING
|
|
&& pm->ps->weaponstate != WEAPON_CHARGING
|
|
&& pm->ps->weaponstate != WEAPON_CHARGING_ALT
|
|
&& (PM_RunningAnim( pm->ps->legsAnim )
|
|
|| PM_WalkingAnim( pm->ps->legsAnim )
|
|
|| PM_JumpingAnim( pm->ps->legsAnim )
|
|
|| PM_SwimmingAnim( pm->ps->legsAnim )) )
|
|
{//running w/1-handed weapon uses full-body anim
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && (pm->ps->weaponstate == WEAPON_CHARGING || pm->ps->weaponstate == WEAPON_CHARGING_ALT) )
|
|
{//player pulling back to throw
|
|
if ( PM_StandingAnim( pm->ps->legsAnim ) )
|
|
{
|
|
PM_SetAnim( pm, SETANIM_LEGS, BOTH_THERMAL_READY, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD );
|
|
}
|
|
else if ( pm->ps->legsAnim == BOTH_THERMAL_READY )
|
|
{//sigh... hold it so pm_footsteps doesn't override
|
|
if ( pm->ps->legsAnimTimer < 100 )
|
|
{
|
|
pm->ps->legsAnimTimer = 100;
|
|
}
|
|
}
|
|
PM_SetAnim( pm, SETANIM_TORSO, BOTH_THERMAL_READY, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD );
|
|
}
|
|
else
|
|
{
|
|
if ( weaponBusy )
|
|
{
|
|
PM_SetAnim( pm, SETANIM_TORSO, TORSO_WEAPONREADY10, SETANIM_FLAG_NORMAL );
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim( pm, SETANIM_TORSO, BOTH_STAND1, SETANIM_FLAG_NORMAL );
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case WP_REPEATER:
|
|
if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_GALAKMECH )
|
|
{//
|
|
if ( pm->gent->alt_fire )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONREADY3,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONREADY1,SETANIM_FLAG_NORMAL);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONREADY3,SETANIM_FLAG_NORMAL);
|
|
}
|
|
break;
|
|
case WP_TRIP_MINE:
|
|
case WP_DET_PACK:
|
|
if ( PM_RunningAnim( pm->ps->legsAnim )
|
|
|| PM_WalkingAnim( pm->ps->legsAnim )
|
|
|| PM_JumpingAnim( pm->ps->legsAnim )
|
|
|| PM_SwimmingAnim( pm->ps->legsAnim ) )
|
|
{//running w/1-handed weapon uses full-body anim
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
if ( weaponBusy )
|
|
{
|
|
PM_SetAnim( pm, SETANIM_TORSO, TORSO_WEAPONREADY3, SETANIM_FLAG_NORMAL );
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim( pm, SETANIM_TORSO, BOTH_STAND1, SETANIM_FLAG_NORMAL );
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONREADY3,SETANIM_FLAG_NORMAL);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if ( pm->ps->weaponstate == WEAPON_IDLE )
|
|
{
|
|
if( pm->ps->legsAnim == BOTH_GUARD_LOOKAROUND1 )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_GUARD_LOOKAROUND1,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_GUARD_IDLE1 )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_GUARD_IDLE1,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_STAND1IDLE1
|
|
|| pm->ps->legsAnim == BOTH_STAND2IDLE1
|
|
|| pm->ps->legsAnim == BOTH_STAND2IDLE2
|
|
|| pm->ps->legsAnim == BOTH_STAND3IDLE1
|
|
|| pm->ps->legsAnim == BOTH_STAND5IDLE1 )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
pm->ps->saberMove = LS_READY;
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_STAND2TO4 )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_STAND2TO4,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_STAND4TO2 )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_STAND4TO2,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_STAND4 )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_STAND4,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_SWIM_IDLE1 )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_SWIM_IDLE1,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if( pm->ps->legsAnim == BOTH_SWIMFORWARD )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_SWIMFORWARD,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if ( PM_InSpecialJump( pm->ps->legsAnim ) )
|
|
{//use legs anim
|
|
//FIXME: or just use whatever's currently playing?
|
|
//PM_SetAnim( pm, SETANIM_TORSO, pm->ps->legsAnim, SETANIM_FLAG_NORMAL );
|
|
}
|
|
else if ( (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()) && pm->ps->torsoAnim == BOTH_BUTTON_HOLD )
|
|
{//using something
|
|
if ( !pm->ps->useTime )
|
|
{//stopped holding it, release
|
|
PM_SetAnim( pm, SETANIM_TORSO, BOTH_BUTTON_RELEASE, SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_HOLD );
|
|
}//else still holding, leave it as it is
|
|
}
|
|
else
|
|
{
|
|
if ( !weaponBusy
|
|
&& pm->ps->weapon != WP_BOWCASTER
|
|
&& pm->ps->weapon != WP_REPEATER
|
|
&& pm->ps->weapon != WP_FLECHETTE
|
|
&& pm->ps->weapon != WP_ROCKET_LAUNCHER
|
|
&& pm->ps->weapon != WP_CONCUSSION
|
|
&& ( PM_RunningAnim( pm->ps->legsAnim )
|
|
|| (PM_WalkingAnim( pm->ps->legsAnim ) && (pm->ps->clientNum < MAX_CLIENTS||PM_ControlledByPlayer()))
|
|
|| PM_JumpingAnim( pm->ps->legsAnim )
|
|
|| PM_SwimmingAnim( pm->ps->legsAnim ) ) )
|
|
{//running w/1-handed or light 2-handed weapon uses full-body anim if you're not using the weapon right now
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
switch ( pm->ps->weapon )
|
|
{
|
|
// ********************************************************
|
|
case WP_SABER: // WP_LIGHTSABER
|
|
// Shouldn't get here, should go to TorsoAnimLightsaber
|
|
break;
|
|
// ********************************************************
|
|
|
|
case WP_BRYAR_PISTOL:
|
|
if ( pm->ps->weaponstate == WEAPON_CHARGING_ALT || weaponBusy )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONREADY2,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if ( PM_RunningAnim( pm->ps->legsAnim )
|
|
|| PM_WalkingAnim( pm->ps->legsAnim )
|
|
|| PM_JumpingAnim( pm->ps->legsAnim )
|
|
|| PM_SwimmingAnim( pm->ps->legsAnim ) )
|
|
{//running w/1-handed weapon uses full-body anim
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONIDLE2,SETANIM_FLAG_NORMAL);
|
|
}
|
|
break;
|
|
case WP_BLASTER_PISTOL:
|
|
if ( pm->gent
|
|
&& pm->gent->weaponModel[1] > 0 )
|
|
{//dual pistols
|
|
if ( weaponBusy )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_GUNSIT1,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if ( PM_RunningAnim( pm->ps->legsAnim )
|
|
|| PM_WalkingAnim( pm->ps->legsAnim )
|
|
|| PM_JumpingAnim( pm->ps->legsAnim )
|
|
|| PM_SwimmingAnim( pm->ps->legsAnim ) )
|
|
{//running w/1-handed weapon uses full-body anim
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_STAND1,SETANIM_FLAG_NORMAL);
|
|
}
|
|
}
|
|
else
|
|
{//single pistols
|
|
if ( pm->ps->weaponstate == WEAPON_CHARGING_ALT || weaponBusy )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONREADY2,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else if ( PM_RunningAnim( pm->ps->legsAnim )
|
|
|| PM_WalkingAnim( pm->ps->legsAnim )
|
|
|| PM_JumpingAnim( pm->ps->legsAnim )
|
|
|| PM_SwimmingAnim( pm->ps->legsAnim ) )
|
|
{//running w/1-handed weapon uses full-body anim
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONIDLE2,SETANIM_FLAG_NORMAL);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case WP_NONE:
|
|
//NOTE: should never get here
|
|
break;
|
|
|
|
case WP_MELEE:
|
|
if ( PM_RunningAnim( pm->ps->legsAnim )
|
|
|| PM_WalkingAnim( pm->ps->legsAnim )
|
|
|| PM_JumpingAnim( pm->ps->legsAnim )
|
|
|| PM_SwimmingAnim( pm->ps->legsAnim ) )
|
|
{//running w/1-handed weapon uses full-body anim
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
if ( pm->gent && pm->gent->client && pm->gent->client->NPC_class == CLASS_RANCOR )
|
|
{//ignore
|
|
}
|
|
else if ( pm->gent && pm->gent->client && !PM_DroidMelee( pm->gent->client->NPC_class ) )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_STAND6,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,BOTH_STAND1,SETANIM_FLAG_NORMAL);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case WP_TUSKEN_STAFF:
|
|
if ( PM_RunningAnim( pm->ps->legsAnim )
|
|
|| PM_WalkingAnim( pm->ps->legsAnim )
|
|
|| PM_JumpingAnim( pm->ps->legsAnim )
|
|
|| PM_SwimmingAnim( pm->ps->legsAnim ) )
|
|
{//running w/1-handed weapon uses full-body anim
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim(pm, SETANIM_TORSO, BOTH_STAND3, SETANIM_FLAG_NORMAL);
|
|
}
|
|
break;
|
|
|
|
case WP_NOGHRI_STICK:
|
|
if ( weaponBusy )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONREADY3,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONIDLE3,SETANIM_FLAG_NORMAL);
|
|
}
|
|
break;
|
|
|
|
case WP_BLASTER:
|
|
if ( weaponBusy )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONREADY3,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONIDLE3,SETANIM_FLAG_NORMAL);
|
|
}
|
|
break;
|
|
|
|
case WP_DISRUPTOR:
|
|
case WP_TUSKEN_RIFLE:
|
|
if ( (pm->ps->weaponstate != WEAPON_FIRING
|
|
&& pm->ps->weaponstate != WEAPON_CHARGING
|
|
&& pm->ps->weaponstate != WEAPON_CHARGING_ALT)
|
|
|| PM_RunningAnim( pm->ps->legsAnim )
|
|
|| PM_WalkingAnim( pm->ps->legsAnim )
|
|
|| PM_JumpingAnim( pm->ps->legsAnim )
|
|
|| PM_SwimmingAnim( pm->ps->legsAnim ) )
|
|
{//running sniper weapon uses normal ready
|
|
if ( pm->ps->clientNum )
|
|
{
|
|
PM_SetAnim( pm, SETANIM_TORSO, TORSO_WEAPONREADY3, SETANIM_FLAG_NORMAL );
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim( pm, SETANIM_TORSO, TORSO_WEAPONREADY3, SETANIM_FLAG_NORMAL );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ( pm->ps->clientNum )
|
|
{
|
|
PM_SetAnim( pm, SETANIM_TORSO, TORSO_WEAPONREADY4, SETANIM_FLAG_NORMAL );
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim( pm, SETANIM_TORSO, TORSO_WEAPONREADY4, SETANIM_FLAG_NORMAL );
|
|
}
|
|
}
|
|
break;
|
|
|
|
case WP_BOT_LASER:
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONIDLE2,SETANIM_FLAG_OVERRIDE|SETANIM_FLAG_RESTART|SETANIM_FLAG_HOLD);
|
|
break;
|
|
|
|
case WP_THERMAL:
|
|
if ( PM_RunningAnim( pm->ps->legsAnim )
|
|
|| PM_WalkingAnim( pm->ps->legsAnim )
|
|
|| PM_JumpingAnim( pm->ps->legsAnim )
|
|
|| PM_SwimmingAnim( pm->ps->legsAnim ) )
|
|
{//running w/1-handed weapon uses full-body anim
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
if ( weaponBusy )
|
|
{
|
|
PM_SetAnim( pm, SETANIM_TORSO, TORSO_WEAPONIDLE10, SETANIM_FLAG_NORMAL );
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim( pm, SETANIM_TORSO, BOTH_STAND1, SETANIM_FLAG_NORMAL );
|
|
}
|
|
}
|
|
break;
|
|
|
|
case WP_REPEATER:
|
|
if ( weaponBusy )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONREADY3,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONIDLE3,SETANIM_FLAG_NORMAL);
|
|
}
|
|
break;
|
|
case WP_TRIP_MINE:
|
|
case WP_DET_PACK:
|
|
if ( PM_RunningAnim( pm->ps->legsAnim )
|
|
|| PM_WalkingAnim( pm->ps->legsAnim )
|
|
|| PM_JumpingAnim( pm->ps->legsAnim )
|
|
|| PM_SwimmingAnim( pm->ps->legsAnim ) )
|
|
{//running w/1-handed weapon uses full-body anim
|
|
PM_SetAnim(pm,SETANIM_TORSO,pm->ps->legsAnim,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
if ( weaponBusy )
|
|
{
|
|
PM_SetAnim( pm, SETANIM_TORSO, TORSO_WEAPONIDLE3, SETANIM_FLAG_NORMAL );
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim( pm, SETANIM_TORSO, BOTH_STAND1, SETANIM_FLAG_NORMAL );
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
if ( weaponBusy )
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONREADY3,SETANIM_FLAG_NORMAL);
|
|
}
|
|
else
|
|
{
|
|
PM_SetAnim(pm,SETANIM_TORSO,TORSO_WEAPONIDLE3,SETANIM_FLAG_NORMAL);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//=========================================================================
|
|
// Anim checking utils
|
|
//=========================================================================
|
|
|
|
int PM_GetTurnAnim( gentity_t *gent, int anim )
|
|
{
|
|
if ( !gent )
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
switch( anim )
|
|
{
|
|
case BOTH_STAND1: //# Standing idle: no weapon: hands down
|
|
case BOTH_STAND1IDLE1: //# Random standing idle
|
|
case BOTH_STAND2: //# Standing idle with a weapon
|
|
case BOTH_SABERFAST_STANCE:
|
|
case BOTH_SABERSLOW_STANCE:
|
|
case BOTH_STAND2IDLE1: //# Random standing idle
|
|
case BOTH_STAND2IDLE2: //# Random standing idle
|
|
case BOTH_STAND3: //# Standing hands behind back: at ease: etc.
|
|
case BOTH_STAND3IDLE1: //# Random standing idle
|
|
case BOTH_STAND4: //# two handed: gun down: relaxed stand
|
|
case BOTH_STAND5: //# standing idle, no weapon, hand down, back straight
|
|
case BOTH_STAND5IDLE1: //# Random standing idle
|
|
case BOTH_STAND6: //# one handed: gun at side: relaxed stand
|
|
case BOTH_STAND2TO4: //# Transition from stand2 to stand4
|
|
case BOTH_STAND4TO2: //# Transition from stand4 to stand2
|
|
case BOTH_GESTURE1: //# Generic gesture: non-specific
|
|
case BOTH_GESTURE2: //# Generic gesture: non-specific
|
|
case BOTH_TALK1: //# Generic talk anim
|
|
case BOTH_TALK2: //# Generic talk anim
|
|
if ( PM_HasAnimation( gent, LEGS_TURN1 ) )
|
|
{
|
|
return LEGS_TURN1;
|
|
}
|
|
else
|
|
{
|
|
return -1;
|
|
}
|
|
break;
|
|
case BOTH_ATTACK1: //# Attack with generic 1-handed weapon
|
|
case BOTH_ATTACK2: //# Attack with generic 2-handed weapon
|
|
case BOTH_ATTACK3: //# Attack with heavy 2-handed weapon
|
|
case BOTH_ATTACK4: //# Attack with ???
|
|
case BOTH_MELEE1: //# First melee attack
|
|
case BOTH_MELEE2: //# Second melee attack
|
|
case BOTH_GUARD_LOOKAROUND1: //# Cradling weapon and looking around
|
|
case BOTH_GUARD_IDLE1: //# Cradling weapon and standing
|
|
if ( PM_HasAnimation( gent, LEGS_TURN2 ) )
|
|
{
|
|
return LEGS_TURN2;
|
|
}
|
|
else
|
|
{
|
|
return -1;
|
|
}
|
|
break;
|
|
default:
|
|
return -1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
int PM_TurnAnimForLegsAnim( gentity_t *gent, int anim )
|
|
{
|
|
if ( !gent )
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
switch( anim )
|
|
{
|
|
case BOTH_STAND1: //# Standing idle: no weapon: hands down
|
|
case BOTH_STAND1IDLE1: //# Random standing idle
|
|
if ( PM_HasAnimation( gent, BOTH_TURNSTAND1 ) )
|
|
{
|
|
return BOTH_TURNSTAND1;
|
|
}
|
|
else
|
|
{
|
|
return -1;
|
|
}
|
|
break;
|
|
case BOTH_STAND2: //# Standing idle with a weapon
|
|
case BOTH_SABERFAST_STANCE:
|
|
case BOTH_SABERSLOW_STANCE:
|
|
case BOTH_STAND2IDLE1: //# Random standing idle
|
|
case BOTH_STAND2IDLE2: //# Random standing idle
|
|
if ( PM_HasAnimation( gent, BOTH_TURNSTAND2 ) )
|
|
{
|
|
return BOTH_TURNSTAND2;
|
|
}
|
|
else
|
|
{
|
|
return -1;
|
|
}
|
|
break;
|
|
case BOTH_STAND3: //# Standing hands behind back: at ease: etc.
|
|
case BOTH_STAND3IDLE1: //# Random standing idle
|
|
if ( PM_HasAnimation( gent, BOTH_TURNSTAND3 ) )
|
|
{
|
|
return BOTH_TURNSTAND3;
|
|
}
|
|
else
|
|
{
|
|
return -1;
|
|
}
|
|
break;
|
|
case BOTH_STAND4: //# two handed: gun down: relaxed stand
|
|
if ( PM_HasAnimation( gent, BOTH_TURNSTAND4 ) )
|
|
{
|
|
return BOTH_TURNSTAND4;
|
|
}
|
|
else
|
|
{
|
|
return -1;
|
|
}
|
|
break;
|
|
case BOTH_STAND5: //# standing idle, no weapon, hand down, back straight
|
|
case BOTH_STAND5IDLE1: //# Random standing idle
|
|
if ( PM_HasAnimation( gent, BOTH_TURNSTAND5 ) )
|
|
{
|
|
return BOTH_TURNSTAND5;
|
|
}
|
|
else
|
|
{
|
|
return -1;
|
|
}
|
|
break;
|
|
case BOTH_CROUCH1: //# Transition from standing to crouch
|
|
case BOTH_CROUCH1IDLE: //# Crouching idle
|
|
/*
|
|
case BOTH_UNCROUCH1: //# Transition from crouch to standing
|
|
case BOTH_CROUCH2TOSTAND1: //# going from crouch2 to stand1
|
|
case BOTH_CROUCH3: //# Desann crouching down to Kyle (cin 9)
|
|
case BOTH_UNCROUCH3: //# Desann uncrouching down to Kyle (cin 9)
|
|
case BOTH_CROUCH4: //# Slower version of crouch1 for cinematics
|
|
case BOTH_UNCROUCH4: //# Slower version of uncrouch1 for cinematics
|
|
*/
|
|
if ( PM_HasAnimation( gent, BOTH_TURNCROUCH1 ) )
|
|
{
|
|
return BOTH_TURNCROUCH1;
|
|
}
|
|
else
|
|
{
|
|
return -1;
|
|
}
|
|
break;
|
|
default:
|
|
return -1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
qboolean PM_InOnGroundAnim ( playerState_t *ps )
|
|
{
|
|
switch( ps->legsAnim )
|
|
{
|
|
case BOTH_DEAD1:
|
|
case BOTH_DEAD2:
|
|
case BOTH_DEAD3:
|
|
case BOTH_DEAD4:
|
|
case BOTH_DEAD5:
|
|
case BOTH_DEADFORWARD1:
|
|
case BOTH_DEADBACKWARD1:
|
|
case BOTH_DEADFORWARD2:
|
|
case BOTH_DEADBACKWARD2:
|
|
case BOTH_LYINGDEATH1:
|
|
case BOTH_LYINGDEAD1:
|
|
case BOTH_SLEEP1: //# laying on back-rknee up-rhand on torso
|
|
return qtrue;
|
|
break;
|
|
case BOTH_KNOCKDOWN1: //#
|
|
case BOTH_KNOCKDOWN2: //#
|
|
case BOTH_KNOCKDOWN3: //#
|
|
case BOTH_KNOCKDOWN4: //#
|
|
case BOTH_KNOCKDOWN5: //#
|
|
case BOTH_LK_DL_ST_T_SB_1_L:
|
|
case BOTH_RELEASED:
|
|
if ( ps->legsAnimTimer < 500 )
|
|
{//pretty much horizontal by this point
|
|
return qtrue;
|
|
}
|
|
break;
|
|
case BOTH_PLAYER_PA_3_FLY:
|
|
if ( ps->legsAnimTimer < 300 )
|
|
{//pretty much horizontal by this point
|
|
return qtrue;
|
|
}
|
|
/*
|
|
else if ( ps->clientNum < MAX_CLIENTS
|
|
&& ps->legsAnimTimer < 300 + PLAYER_KNOCKDOWN_HOLD_EXTRA_TIME )
|
|
{
|
|
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 ( ps->legsAnimTimer > PM_AnimLength( g_entities[ps->clientNum].client->clientInfo.animFileIndex, (animNumber_t)ps->legsAnim )-400 )
|
|
{//still pretty much horizontal at this point
|
|
return qtrue;
|
|
}
|
|
break;
|
|
}
|
|
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean PM_InSpecialDeathAnim( int anim )
|
|
{
|
|
switch( pm->ps->legsAnim )
|
|
{
|
|
case BOTH_DEATH_ROLL: //# Death anim from a roll
|
|
case BOTH_DEATH_FLIP: //# Death anim from a flip
|
|
case BOTH_DEATH_SPIN_90_R: //# Death anim when facing 90 degrees right
|
|
case BOTH_DEATH_SPIN_90_L: //# Death anim when facing 90 degrees left
|
|
case BOTH_DEATH_SPIN_180: //# Death anim when facing backwards
|
|
case BOTH_DEATH_LYING_UP: //# Death anim when lying on back
|
|
case BOTH_DEATH_LYING_DN: //# Death anim when lying on front
|
|
case BOTH_DEATH_FALLING_DN: //# Death anim when falling on face
|
|
case BOTH_DEATH_FALLING_UP: //# Death anim when falling on back
|
|
case BOTH_DEATH_CROUCHED: //# Death anim when crouched
|
|
return qtrue;
|
|
break;
|
|
default:
|
|
return qfalse;
|
|
break;
|
|
}
|
|
}
|
|
|
|
qboolean PM_InDeathAnim ( void )
|
|
{//Purposely does not cover stumbledeath and falldeath...
|
|
switch( pm->ps->legsAnim )
|
|
{
|
|
case BOTH_DEATH1: //# First Death anim
|
|
case BOTH_DEATH2: //# Second Death anim
|
|
case BOTH_DEATH3: //# Third Death anim
|
|
case BOTH_DEATH4: //# Fourth Death anim
|
|
case BOTH_DEATH5: //# Fifth Death anim
|
|
case BOTH_DEATH6: //# Sixth Death anim
|
|
case BOTH_DEATH7: //# Seventh Death anim
|
|
case BOTH_DEATH8: //#
|
|
case BOTH_DEATH9: //#
|
|
case BOTH_DEATH10: //#
|
|
case BOTH_DEATH11: //#
|
|
case BOTH_DEATH12: //#
|
|
case BOTH_DEATH13: //#
|
|
case BOTH_DEATH14: //#
|
|
case BOTH_DEATH14_UNGRIP: //# Desann's end death (cin #35)
|
|
case BOTH_DEATH14_SITUP: //# Tavion sitting up after having been thrown (cin #23)
|
|
case BOTH_DEATH15: //#
|
|
case BOTH_DEATH16: //#
|
|
case BOTH_DEATH17: //#
|
|
case BOTH_DEATH18: //#
|
|
case BOTH_DEATH19: //#
|
|
case BOTH_DEATH20: //#
|
|
case BOTH_DEATH21: //#
|
|
case BOTH_DEATH22: //#
|
|
case BOTH_DEATH23: //#
|
|
case BOTH_DEATH24: //#
|
|
case BOTH_DEATH25: //#
|
|
|
|
case BOTH_DEATHFORWARD1: //# First Death in which they get thrown forward
|
|
case BOTH_DEATHFORWARD2: //# Second Death in which they get thrown forward
|
|
case BOTH_DEATHFORWARD3: //# Tavion's falling in cin# 23
|
|
case BOTH_DEATHBACKWARD1: //# First Death in which they get thrown backward
|
|
case BOTH_DEATHBACKWARD2: //# Second Death in which they get thrown backward
|
|
|
|
case BOTH_DEATH1IDLE: //# Idle while close to death
|
|
case BOTH_LYINGDEATH1: //# Death to play when killed lying down
|
|
case BOTH_STUMBLEDEATH1: //# Stumble forward and fall face first death
|
|
case BOTH_FALLDEATH1: //# Fall forward off a high cliff and splat death - start
|
|
case BOTH_FALLDEATH1INAIR: //# Fall forward off a high cliff and splat death - loop
|
|
case BOTH_FALLDEATH1LAND: //# Fall forward off a high cliff and splat death - hit bottom
|
|
//# #sep case BOTH_ DEAD POSES # Should be last frame of corresponding previous anims
|
|
case BOTH_DEAD1: //# First Death finished pose
|
|
case BOTH_DEAD2: //# Second Death finished pose
|
|
case BOTH_DEAD3: //# Third Death finished pose
|
|
case BOTH_DEAD4: //# Fourth Death finished pose
|
|
case BOTH_DEAD5: //# Fifth Death finished pose
|
|
case BOTH_DEAD6: //# Sixth Death finished pose
|
|
case BOTH_DEAD7: //# Seventh Death finished pose
|
|
case BOTH_DEAD8: //#
|
|
case BOTH_DEAD9: //#
|
|
case BOTH_DEAD10: //#
|
|
case BOTH_DEAD11: //#
|
|
case BOTH_DEAD12: //#
|
|
case BOTH_DEAD13: //#
|
|
case BOTH_DEAD14: //#
|
|
case BOTH_DEAD15: //#
|
|
case BOTH_DEAD16: //#
|
|
case BOTH_DEAD17: //#
|
|
case BOTH_DEAD18: //#
|
|
case BOTH_DEAD19: //#
|
|
case BOTH_DEAD20: //#
|
|
case BOTH_DEAD21: //#
|
|
case BOTH_DEAD22: //#
|
|
case BOTH_DEAD23: //#
|
|
case BOTH_DEAD24: //#
|
|
case BOTH_DEAD25: //#
|
|
case BOTH_DEADFORWARD1: //# First thrown forward death finished pose
|
|
case BOTH_DEADFORWARD2: //# Second thrown forward death finished pose
|
|
case BOTH_DEADBACKWARD1: //# First thrown backward death finished pose
|
|
case BOTH_DEADBACKWARD2: //# Second thrown backward death finished pose
|
|
case BOTH_LYINGDEAD1: //# Killed lying down death finished pose
|
|
case BOTH_STUMBLEDEAD1: //# Stumble forward death finished pose
|
|
case BOTH_FALLDEAD1LAND: //# Fall forward and splat death finished pose
|
|
//# #sep case BOTH_ DEAD TWITCH/FLOP # React to being shot from death poses
|
|
case BOTH_DEADFLOP1: //# React to being shot from First Death finished pose
|
|
case BOTH_DEADFLOP2: //# React to being shot from Second Death finished pose
|
|
case BOTH_DISMEMBER_HEAD1: //#
|
|
case BOTH_DISMEMBER_TORSO1: //#
|
|
case BOTH_DISMEMBER_LLEG: //#
|
|
case BOTH_DISMEMBER_RLEG: //#
|
|
case BOTH_DISMEMBER_RARM: //#
|
|
case BOTH_DISMEMBER_LARM: //#
|
|
return qtrue;
|
|
break;
|
|
default:
|
|
return PM_InSpecialDeathAnim( pm->ps->legsAnim );
|
|
break;
|
|
}
|
|
}
|
|
|
|
qboolean PM_InCartwheel( int anim )
|
|
{
|
|
switch ( anim )
|
|
{
|
|
case BOTH_ARIAL_LEFT:
|
|
case BOTH_ARIAL_RIGHT:
|
|
case BOTH_ARIAL_F1:
|
|
case BOTH_CARTWHEEL_LEFT:
|
|
case BOTH_CARTWHEEL_RIGHT:
|
|
return qtrue;
|
|
break;
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean PM_InButterfly( int anim )
|
|
{
|
|
switch ( anim )
|
|
{
|
|
case BOTH_BUTTERFLY_LEFT:
|
|
case BOTH_BUTTERFLY_RIGHT:
|
|
case BOTH_BUTTERFLY_FL1:
|
|
case BOTH_BUTTERFLY_FR1:
|
|
return qtrue;
|
|
break;
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean PM_StandingAnim( int anim )
|
|
{//NOTE: does not check idles or special (cinematic) stands
|
|
switch ( anim )
|
|
{
|
|
case BOTH_STAND1:
|
|
case BOTH_STAND2:
|
|
case BOTH_STAND3:
|
|
case BOTH_STAND4:
|
|
case BOTH_ATTACK3:
|
|
return qtrue;
|
|
break;
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean PM_InAirKickingAnim( int anim )
|
|
{
|
|
switch ( anim )
|
|
{
|
|
case BOTH_A7_KICK_F_AIR:
|
|
case BOTH_A7_KICK_B_AIR:
|
|
case BOTH_A7_KICK_R_AIR:
|
|
case BOTH_A7_KICK_L_AIR:
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean PM_KickingAnim( int anim )
|
|
{
|
|
switch ( anim )
|
|
{
|
|
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:
|
|
//NOT a kick, but acts like one:
|
|
case BOTH_A7_HILT:
|
|
//NOT kicks, but do kick traces anyway
|
|
case BOTH_GETUP_BROLL_B:
|
|
case BOTH_GETUP_BROLL_F:
|
|
case BOTH_GETUP_FROLL_B:
|
|
case BOTH_GETUP_FROLL_F:
|
|
return qtrue;
|
|
break;
|
|
default:
|
|
return PM_InAirKickingAnim( anim );
|
|
break;
|
|
}
|
|
//return qfalse;
|
|
}
|
|
|
|
qboolean PM_StabDownAnim( int anim )
|
|
{
|
|
switch ( anim )
|
|
{
|
|
case BOTH_STABDOWN:
|
|
case BOTH_STABDOWN_STAFF:
|
|
case BOTH_STABDOWN_DUAL:
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean PM_GoingToAttackDown( playerState_t *ps )
|
|
{
|
|
if ( PM_StabDownAnim( ps->torsoAnim )//stabbing downward
|
|
|| ps->saberMove == LS_A_LUNGE//lunge
|
|
|| ps->saberMove == LS_A_JUMP_T__B_//death from above
|
|
|| ps->saberMove == LS_A_T2B//attacking top to bottom
|
|
|| ps->saberMove == LS_S_T2B//starting at attack downward
|
|
|| (PM_SaberInTransition( ps->saberMove ) && saberMoveData[ps->saberMove].endQuad == Q_T) )//transitioning to a top to bottom attack
|
|
{
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean PM_ForceUsingSaberAnim( int anim )
|
|
{//saber/acrobatic anims that should prevent you from recharging force power while you're in them...
|
|
switch ( anim )
|
|
{
|
|
case BOTH_JUMPFLIPSLASHDOWN1:
|
|
case BOTH_JUMPFLIPSTABDOWN:
|
|
case BOTH_FORCELEAP2_T__B_:
|
|
case BOTH_JUMPATTACK6:
|
|
case BOTH_JUMPATTACK7:
|
|
case BOTH_FORCELONGLEAP_START:
|
|
case BOTH_FORCELONGLEAP_ATTACK:
|
|
case BOTH_FORCEWALLRUNFLIP_START:
|
|
case BOTH_FORCEWALLRUNFLIP_END:
|
|
case BOTH_FORCEWALLRUNFLIP_ALT:
|
|
case BOTH_FORCEWALLREBOUND_FORWARD:
|
|
case BOTH_FORCEWALLREBOUND_LEFT:
|
|
case BOTH_FORCEWALLREBOUND_BACK:
|
|
case BOTH_FORCEWALLREBOUND_RIGHT:
|
|
case BOTH_FLIP_ATTACK7:
|
|
case BOTH_FLIP_HOLD7:
|
|
case BOTH_FLIP_LAND:
|
|
case BOTH_PULL_IMPALE_STAB:
|
|
case BOTH_PULL_IMPALE_SWING:
|
|
case BOTH_A6_SABERPROTECT:
|
|
case BOTH_A7_SOULCAL:
|
|
case BOTH_A1_SPECIAL:
|
|
case BOTH_A2_SPECIAL:
|
|
case BOTH_A3_SPECIAL:
|
|
case BOTH_ARIAL_LEFT:
|
|
case BOTH_ARIAL_RIGHT:
|
|
case BOTH_CARTWHEEL_LEFT:
|
|
case BOTH_CARTWHEEL_RIGHT:
|
|
case BOTH_FLIP_LEFT:
|
|
case BOTH_FLIP_BACK1:
|
|
case BOTH_FLIP_BACK2:
|
|
case BOTH_FLIP_BACK3:
|
|
case BOTH_ALORA_FLIP_B:
|
|
case BOTH_BUTTERFLY_LEFT:
|
|
case BOTH_BUTTERFLY_RIGHT:
|
|
case BOTH_BUTTERFLY_FL1:
|
|
case BOTH_BUTTERFLY_FR1:
|
|
case BOTH_WALL_RUN_RIGHT:
|
|
case BOTH_WALL_RUN_RIGHT_FLIP:
|
|
case BOTH_WALL_RUN_RIGHT_STOP:
|
|
case BOTH_WALL_RUN_LEFT:
|
|
case BOTH_WALL_RUN_LEFT_FLIP:
|
|
case BOTH_WALL_RUN_LEFT_STOP:
|
|
case BOTH_WALL_FLIP_RIGHT:
|
|
case BOTH_WALL_FLIP_LEFT:
|
|
case BOTH_FORCEJUMP1:
|
|
case BOTH_FORCEINAIR1:
|
|
case BOTH_FORCELAND1:
|
|
case BOTH_FORCEJUMPBACK1:
|
|
case BOTH_FORCEINAIRBACK1:
|
|
case BOTH_FORCELANDBACK1:
|
|
case BOTH_FORCEJUMPLEFT1:
|
|
case BOTH_FORCEINAIRLEFT1:
|
|
case BOTH_FORCELANDLEFT1:
|
|
case BOTH_FORCEJUMPRIGHT1:
|
|
case BOTH_FORCEINAIRRIGHT1:
|
|
case BOTH_FORCELANDRIGHT1:
|
|
case BOTH_FLIP_F:
|
|
case BOTH_FLIP_B:
|
|
case BOTH_FLIP_L:
|
|
case BOTH_FLIP_R:
|
|
case BOTH_ALORA_FLIP_1:
|
|
case BOTH_ALORA_FLIP_2:
|
|
case BOTH_ALORA_FLIP_3:
|
|
case BOTH_DODGE_FL:
|
|
case BOTH_DODGE_FR:
|
|
case BOTH_DODGE_BL:
|
|
case BOTH_DODGE_BR:
|
|
case BOTH_DODGE_L:
|
|
case BOTH_DODGE_R:
|
|
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:
|
|
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:
|
|
case BOTH_WALL_FLIP_BACK1:
|
|
case BOTH_WALL_FLIP_BACK2:
|
|
case BOTH_SPIN1:
|
|
case BOTH_FJSS_TR_BL:
|
|
case BOTH_FJSS_TL_BR:
|
|
case BOTH_DEFLECTSLASH__R__L_FIN:
|
|
case BOTH_ARIAL_F1:
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean G_HasKnockdownAnims( gentity_t *ent )
|
|
{
|
|
if ( PM_HasAnimation( ent, BOTH_KNOCKDOWN1 )
|
|
&& PM_HasAnimation( ent, BOTH_KNOCKDOWN2 )
|
|
&& PM_HasAnimation( ent, BOTH_KNOCKDOWN3 )
|
|
&& PM_HasAnimation( ent, BOTH_KNOCKDOWN4 )
|
|
&& PM_HasAnimation( ent, BOTH_KNOCKDOWN5 ) )
|
|
{
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean PM_InAttackRoll( int anim )
|
|
{
|
|
switch ( anim )
|
|
{
|
|
case BOTH_GETUP_BROLL_B:
|
|
case BOTH_GETUP_BROLL_F:
|
|
case BOTH_GETUP_FROLL_B:
|
|
case BOTH_GETUP_FROLL_F:
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean PM_LockedAnim( int anim )
|
|
{//anims that can *NEVER* be overridden, regardless
|
|
switch ( anim )
|
|
{
|
|
case BOTH_KYLE_PA_1:
|
|
case BOTH_KYLE_PA_2:
|
|
case BOTH_KYLE_PA_3:
|
|
case BOTH_PLAYER_PA_1:
|
|
case BOTH_PLAYER_PA_2:
|
|
case BOTH_PLAYER_PA_3:
|
|
case BOTH_PLAYER_PA_3_FLY:
|
|
case BOTH_TAVION_SCEPTERGROUND:
|
|
case BOTH_TAVION_SWORDPOWER:
|
|
case BOTH_SCEPTER_START:
|
|
case BOTH_SCEPTER_HOLD:
|
|
case BOTH_SCEPTER_STOP:
|
|
//grabbed by wampa
|
|
case BOTH_GRABBED: //#
|
|
case BOTH_RELEASED: //# when Wampa drops player, transitions into fall on back
|
|
case BOTH_HANG_IDLE: //#
|
|
case BOTH_HANG_ATTACK: //#
|
|
case BOTH_HANG_PAIN: //#
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean PM_SuperBreakLoseAnim( int anim )
|
|
{
|
|
switch ( anim )
|
|
{
|
|
case BOTH_LK_S_DL_S_SB_1_L: //super break I lost
|
|
case BOTH_LK_S_DL_T_SB_1_L: //super break I lost
|
|
case BOTH_LK_S_ST_S_SB_1_L: //super break I lost
|
|
case BOTH_LK_S_ST_T_SB_1_L: //super break I lost
|
|
case BOTH_LK_S_S_S_SB_1_L: //super break I lost
|
|
case BOTH_LK_S_S_T_SB_1_L: //super break I lost
|
|
case BOTH_LK_DL_DL_S_SB_1_L: //super break I lost
|
|
case BOTH_LK_DL_DL_T_SB_1_L: //super break I lost
|
|
case BOTH_LK_DL_ST_S_SB_1_L: //super break I lost
|
|
case BOTH_LK_DL_ST_T_SB_1_L: //super break I lost
|
|
case BOTH_LK_DL_S_S_SB_1_L: //super break I lost
|
|
case BOTH_LK_DL_S_T_SB_1_L: //super break I lost
|
|
case BOTH_LK_ST_DL_S_SB_1_L: //super break I lost
|
|
case BOTH_LK_ST_DL_T_SB_1_L: //super break I lost
|
|
case BOTH_LK_ST_ST_S_SB_1_L: //super break I lost
|
|
case BOTH_LK_ST_ST_T_SB_1_L: //super break I lost
|
|
case BOTH_LK_ST_S_S_SB_1_L: //super break I lost
|
|
case BOTH_LK_ST_S_T_SB_1_L: //super break I lost
|
|
return qtrue;
|
|
break;
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean PM_SuperBreakWinAnim( int anim )
|
|
{
|
|
switch ( anim )
|
|
{
|
|
case BOTH_LK_S_DL_S_SB_1_W: //super break I won
|
|
case BOTH_LK_S_DL_T_SB_1_W: //super break I won
|
|
case BOTH_LK_S_ST_S_SB_1_W: //super break I won
|
|
case BOTH_LK_S_ST_T_SB_1_W: //super break I won
|
|
case BOTH_LK_S_S_S_SB_1_W: //super break I won
|
|
case BOTH_LK_S_S_T_SB_1_W: //super break I won
|
|
case BOTH_LK_DL_DL_S_SB_1_W: //super break I won
|
|
case BOTH_LK_DL_DL_T_SB_1_W: //super break I won
|
|
case BOTH_LK_DL_ST_S_SB_1_W: //super break I won
|
|
case BOTH_LK_DL_ST_T_SB_1_W: //super break I won
|
|
case BOTH_LK_DL_S_S_SB_1_W: //super break I won
|
|
case BOTH_LK_DL_S_T_SB_1_W: //super break I won
|
|
case BOTH_LK_ST_DL_S_SB_1_W: //super break I won
|
|
case BOTH_LK_ST_DL_T_SB_1_W: //super break I won
|
|
case BOTH_LK_ST_ST_S_SB_1_W: //super break I won
|
|
case BOTH_LK_ST_ST_T_SB_1_W: //super break I won
|
|
case BOTH_LK_ST_S_S_SB_1_W: //super break I won
|
|
case BOTH_LK_ST_S_T_SB_1_W: //super break I won
|
|
return qtrue;
|
|
break;
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean PM_SaberLockBreakAnim( int anim )
|
|
{
|
|
switch ( anim )
|
|
{
|
|
case BOTH_BF1BREAK:
|
|
case BOTH_BF2BREAK:
|
|
case BOTH_CWCIRCLEBREAK:
|
|
case BOTH_CCWCIRCLEBREAK:
|
|
case BOTH_LK_S_DL_S_B_1_L: //normal break I lost
|
|
case BOTH_LK_S_DL_S_B_1_W: //normal break I won
|
|
case BOTH_LK_S_DL_T_B_1_L: //normal break I lost
|
|
case BOTH_LK_S_DL_T_B_1_W: //normal break I won
|
|
case BOTH_LK_S_ST_S_B_1_L: //normal break I lost
|
|
case BOTH_LK_S_ST_S_B_1_W: //normal break I won
|
|
case BOTH_LK_S_ST_T_B_1_L: //normal break I lost
|
|
case BOTH_LK_S_ST_T_B_1_W: //normal break I won
|
|
case BOTH_LK_S_S_S_B_1_L: //normal break I lost
|
|
case BOTH_LK_S_S_S_B_1_W: //normal break I won
|
|
case BOTH_LK_S_S_T_B_1_L: //normal break I lost
|
|
case BOTH_LK_S_S_T_B_1_W: //normal break I won
|
|
case BOTH_LK_DL_DL_S_B_1_L: //normal break I lost
|
|
case BOTH_LK_DL_DL_S_B_1_W: //normal break I won
|
|
case BOTH_LK_DL_DL_T_B_1_L: //normal break I lost
|
|
case BOTH_LK_DL_DL_T_B_1_W: //normal break I won
|
|
case BOTH_LK_DL_ST_S_B_1_L: //normal break I lost
|
|
case BOTH_LK_DL_ST_S_B_1_W: //normal break I won
|
|
case BOTH_LK_DL_ST_T_B_1_L: //normal break I lost
|
|
case BOTH_LK_DL_ST_T_B_1_W: //normal break I won
|
|
case BOTH_LK_DL_S_S_B_1_L: //normal break I lost
|
|
case BOTH_LK_DL_S_S_B_1_W: //normal break I won
|
|
case BOTH_LK_DL_S_T_B_1_L: //normal break I lost
|
|
case BOTH_LK_DL_S_T_B_1_W: //normal break I won
|
|
case BOTH_LK_ST_DL_S_B_1_L: //normal break I lost
|
|
case BOTH_LK_ST_DL_S_B_1_W: //normal break I won
|
|
case BOTH_LK_ST_DL_T_B_1_L: //normal break I lost
|
|
case BOTH_LK_ST_DL_T_B_1_W: //normal break I won
|
|
case BOTH_LK_ST_ST_S_B_1_L: //normal break I lost
|
|
case BOTH_LK_ST_ST_S_B_1_W: //normal break I won
|
|
case BOTH_LK_ST_ST_T_B_1_L: //normal break I lost
|
|
case BOTH_LK_ST_ST_T_B_1_W: //normal break I won
|
|
case BOTH_LK_ST_S_S_B_1_L: //normal break I lost
|
|
case BOTH_LK_ST_S_S_B_1_W: //normal break I won
|
|
case BOTH_LK_ST_S_T_B_1_L: //normal break I lost
|
|
case BOTH_LK_ST_S_T_B_1_W: //normal break I won
|
|
return (PM_SuperBreakLoseAnim(anim)||PM_SuperBreakWinAnim(anim));
|
|
break;
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean PM_GetupAnimNoMove( int legsAnim )
|
|
{
|
|
switch( 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:
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean PM_KnockDownAnim( int anim )
|
|
{
|
|
switch ( anim )
|
|
{
|
|
case BOTH_KNOCKDOWN1:
|
|
case BOTH_KNOCKDOWN2:
|
|
case BOTH_KNOCKDOWN3:
|
|
case BOTH_KNOCKDOWN4:
|
|
case BOTH_KNOCKDOWN5:
|
|
/*
|
|
//special anims:
|
|
case BOTH_RELEASED:
|
|
case BOTH_LK_DL_ST_T_SB_1_L:
|
|
case BOTH_PLAYER_PA_3_FLY:
|
|
*/
|
|
return qtrue;
|
|
break;
|
|
/*
|
|
default:
|
|
return PM_InGetUp( ps );
|
|
break;
|
|
*/
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean PM_KnockDownAnimExtended( int anim )
|
|
{
|
|
switch ( anim )
|
|
{
|
|
case BOTH_KNOCKDOWN1:
|
|
case BOTH_KNOCKDOWN2:
|
|
case BOTH_KNOCKDOWN3:
|
|
case BOTH_KNOCKDOWN4:
|
|
case BOTH_KNOCKDOWN5:
|
|
//special anims:
|
|
case BOTH_RELEASED:
|
|
case BOTH_LK_DL_ST_T_SB_1_L:
|
|
case BOTH_PLAYER_PA_3_FLY:
|
|
return qtrue;
|
|
break;
|
|
/*
|
|
default:
|
|
return PM_InGetUp( ps );
|
|
break;
|
|
*/
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean PM_SaberInKata( saberMoveName_t saberMove )
|
|
{
|
|
switch ( saberMove )
|
|
{
|
|
case LS_A1_SPECIAL:
|
|
case LS_A2_SPECIAL:
|
|
case LS_A3_SPECIAL:
|
|
case LS_DUAL_SPIN_PROTECT:
|
|
case LS_STAFF_SOULCAL:
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean PM_CanRollFromSoulCal( playerState_t *ps )
|
|
{
|
|
if ( ps->legsAnim == BOTH_A7_SOULCAL
|
|
&& ps->legsAnimTimer < 700
|
|
&& ps->legsAnimTimer > 250 )
|
|
{
|
|
return qtrue;
|
|
}
|
|
return qfalse;
|
|
}
|
|
|
|
qboolean BG_FullBodyTauntAnim( int anim )
|
|
{
|
|
switch ( anim )
|
|
{
|
|
case BOTH_GESTURE1:
|
|
case BOTH_DUAL_TAUNT:
|
|
case BOTH_STAFF_TAUNT:
|
|
case BOTH_BOW:
|
|
case BOTH_MEDITATE:
|
|
case BOTH_SHOWOFF_FAST:
|
|
case BOTH_SHOWOFF_MEDIUM:
|
|
case BOTH_SHOWOFF_STRONG:
|
|
case BOTH_SHOWOFF_DUAL:
|
|
case BOTH_SHOWOFF_STAFF:
|
|
case BOTH_VICTORY_FAST:
|
|
case BOTH_VICTORY_MEDIUM:
|
|
case BOTH_VICTORY_STRONG:
|
|
case BOTH_VICTORY_DUAL:
|
|
case BOTH_VICTORY_STAFF:
|
|
return qtrue;
|
|
break;
|
|
}
|
|
return qfalse;
|
|
}
|