/* server/player/player_core.qc Various stuff done for the player, including per-frame functions like PlayerPreThink and PlayerPostThink, also client specific stuff like PutClientInServer etc. Copyright (C) 2021-2024 NZ:P Team This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to: Free Software Foundation, Inc. 59 Temple Place - Suite 330 Boston, MA 02111-1307, USA */ void(entity e) Light_None; void() Spawns_Init; void() SUB_UseTargets; void(entity client) LastStand_UnlinkRevivee; #define PLAYER_START_HEALTH 100 #define PLAYER_CROUCH_DIFFERENCE_HL 25 #define PLAYER_PRONE_DIFFERENCE_HL 23 #define PLAYER_CROUCH_DIFFERENCE_QK 15 #define PLAYER_PRONE_DIFFERENCE_QK 13 #define PLAYER_ANIM_WALK 1 #define PLAYER_ANIM_SPRINT 2 // // Player 3rd Person Animations // // Walking void() PAnim_Walk =[ 1, PAnim_Walk1 ] {self.frame = 0; self.tp_anim_time = time + 0.5;} void() PAnim_Walk1 =[ 2, PAnim_Walk2 ] {self.frame = 1;} void() PAnim_Walk2 =[ 3, PAnim_Walk3 ] {self.frame = 2;} void() PAnim_Walk3 =[ 4, PAnim_Walk4 ] {self.frame = 3;} void() PAnim_Walk4 =[ 5, PAnim_Walk5 ] {self.frame = 4;} void() PAnim_Walk5 =[ 6, PAnim_Walk6 ] {self.frame = 5;} void() PAnim_Walk6 =[ 7, PAnim_Walk7 ] {self.frame = 6;} void() PAnim_Walk7 =[ 8, PAnim_Walk8 ] {self.frame = 7;} void() PAnim_Walk8 =[ 9, SUB_Null ] {self.frame = 8;} // Sprinting void() PAnim_Sprint =[ 1, PAnim_Sprint1 ] {self.frame = 25; self.tp_anim_time = time + 0.65;} void() PAnim_Sprint1 =[ 2, PAnim_Sprint2 ] {self.frame = 26;} void() PAnim_Sprint2 =[ 3, PAnim_Sprint3 ] {self.frame = 27;} void() PAnim_Sprint3 =[ 4, PAnim_Sprint4 ] {self.frame = 28;} void() PAnim_Sprint4 =[ 5, PAnim_Sprint5 ] {self.frame = 29;} void() PAnim_Sprint5 =[ 6, PAnim_Sprint6 ] {self.frame = 30;} void() PAnim_Sprint6 =[ 7, SUB_Null ] {self.frame = 31;} // Reloading void() PAnim_Reload =[ 1, PAnim_Reload1 ] {self.frame = 11; self.tp_anim_time = time + 1;} void() PAnim_Reload1 =[ 2, PAnim_Reload2 ] {self.frame = 12;} void() PAnim_Reload2 =[ 3, PAnim_Reload3 ] {self.frame = 13;} void() PAnim_Reload3 =[ 4, PAnim_Reload4 ] {self.frame = 14;} void() PAnim_Reload4 =[ 5, PAnim_Reload5 ] {self.frame = 15;} void() PAnim_Reload5 =[ 6, PAnim_Reload6 ] {self.frame = 16;} void() PAnim_Reload6 =[ 7, PAnim_Reload7 ] {self.frame = 17;} void() PAnim_Reload7 =[ 8, PAnim_Reload8 ] {self.frame = 18;} void() PAnim_Reload8 =[ 9, PAnim_Reload9 ] {self.frame = 19;} void() PAnim_Reload9 =[ 10, PAnim_Reload10 ] {self.frame = 20;} void() PAnim_Reload10 =[ 11, PAnim_Reload11 ] {self.frame = 21;} void() PAnim_Reload11 =[ 12, PAnim_Reload12 ] {self.frame = 22;} void() PAnim_Reload12 =[ 13, PAnim_Reload13 ] {self.frame = 23;} void() PAnim_Reload13 =[ 14, SUB_Null ] {self.frame = 24;} // Firing void() PAnim_Fire =[ 1, PAnim_Fire1 ] {self.frame = 9; self.tp_anim_time = time + 0.25;} void() PAnim_Fire1 =[ 2, SUB_Null ] {self.frame = 10;} // Melee void() PAnim_Melee =[ 1, PAnim_Melee1 ] {self.frame = 49; self.tp_anim_time = time + 0.65;} void() PAnim_Melee1 =[ 2, PAnim_Melee2 ] {self.frame = 50;} void() PAnim_Melee2 =[ 3, PAnim_Melee3 ] {self.frame = 51;} void() PAnim_Melee3 =[ 4, PAnim_Melee4 ] {self.frame = 52;} void() PAnim_Melee4 =[ 5, PAnim_Melee5 ] {self.frame = 53;} void() PAnim_Melee5 =[ 6, PAnim_Melee6 ] {self.frame = 54;} void() PAnim_Melee6 =[ 7, SUB_Null ] {self.frame = 55;} // Weapon Swap void() PAnim_Swap =[ 1, PAnim_Swap1 ] {self.frame = 56; self.tp_anim_time = time + 1;} void() PAnim_Swap1 =[ 2, PAnim_Swap2 ] {self.frame = 57;} void() PAnim_Swap2 =[ 3, PAnim_Swap3 ] {self.frame = 58;} void() PAnim_Swap3 =[ 4, PAnim_Swap4 ] {self.frame = 59;} void() PAnim_Swap4 =[ 5, PAnim_Swap5 ] {self.frame = 60;} void() PAnim_Swap5 =[ 6, PAnim_Swap6 ] {self.frame = 61;} void() PAnim_Swap6 =[ 7, PAnim_Swap7 ] {self.frame = 62;} void() PAnim_Swap7 =[ 8, PAnim_Swap8 ] {self.frame = 63;} void() PAnim_Swap8 =[ 9, SUB_Null ] {self.frame = 64;} // Enter Dolphin Dive void() PAnim_EnterDive =[ 1, PAnim_EnterDive1 ] {self.frame = 203; self.tp_anim_time = time + 5;} void() PAnim_EnterDive1 =[ 2, PAnim_EnterDive2 ] {self.frame = 204;} void() PAnim_EnterDive2 =[ 3, PAnim_EnterDive3 ] {self.frame = 205;} void() PAnim_EnterDive3 =[ 4, PAnim_EnterDive4 ] {self.frame = 206;} void() PAnim_EnterDive4 =[ 5, SUB_Null ] {self.frame = 207;} // Flop from Dive void() PAnim_Flop =[ 1, PAnim_Flop1 ] {self.frame = 208; self.tp_anim_time = time + 1; nzp_rumble(self, 1000, 1200, 75);} void() PAnim_Flop1 =[ 2, PAnim_Flop2 ] {self.frame = 209;} void() PAnim_Flop2 =[ 3, PAnim_Flop3 ] {self.frame = 210;} void() PAnim_Flop3 =[ 4, SUB_Null ] {self.frame = 211;} // Enter Stand void() PAnim_Stand =[ 1, PAnim_Stand1 ] {self.frame = 114; self.tp_anim_time = time + 1;} void() PAnim_Stand1 =[ 2, PAnim_Stand2 ] {self.frame = 113;} void() PAnim_Stand2 =[ 3, SUB_Null ] {self.frame = 0;} // Enter Crouch void() PAnim_Crouch =[ 1, PAnim_Crouch1 ] {self.frame = 113; self.tp_anim_time = time + 0.5;} void() PAnim_Crouch1 =[ 2, PAnim_Crouch2 ] {self.frame = 114;} void() PAnim_Crouch2 =[ 3, SUB_Null ] {self.frame = 115;} // Walking, while Crouch void() PAnim_CrouchWalk =[ 1, PAnim_CrouchWalk1 ] {self.frame = 116; self.tp_anim_time = time + 0.5;} void() PAnim_CrouchWalk1=[ 2, PAnim_CrouchWalk2 ] {self.frame = 117;} void() PAnim_CrouchWalk2=[ 3, PAnim_CrouchWalk3 ] {self.frame = 118;} void() PAnim_CrouchWalk3=[ 4, PAnim_CrouchWalk4 ] {self.frame = 119;} void() PAnim_CrouchWalk4=[ 5, PAnim_CrouchWalk5 ] {self.frame = 120;} void() PAnim_CrouchWalk5=[ 6, PAnim_CrouchWalk6 ] {self.frame = 121;} void() PAnim_CrouchWalk6=[ 7, PAnim_CrouchWalk7 ] {self.frame = 122;} void() PAnim_CrouchWalk7=[ 8, PAnim_CrouchWalk8 ] {self.frame = 123;} void() PAnim_CrouchWalk8=[ 9, SUB_Null ] {self.frame = 124;} // Reloading, while Crouch void() PAnim_CrReload =[ 1, PAnim_CrReload1 ] {self.frame = 128; self.tp_anim_time = time + 1;} void() PAnim_CrReload1 =[ 2, PAnim_CrReload2 ] {self.frame = 129;} void() PAnim_CrReload2 =[ 3, PAnim_CrReload3 ] {self.frame = 130;} void() PAnim_CrReload3 =[ 4, PAnim_CrReload4 ] {self.frame = 131;} void() PAnim_CrReload4 =[ 5, PAnim_CrReload5 ] {self.frame = 132;} void() PAnim_CrReload5 =[ 6, PAnim_CrReload6 ] {self.frame = 133;} void() PAnim_CrReload6 =[ 7, PAnim_CrReload7 ] {self.frame = 134;} void() PAnim_CrReload7 =[ 8, PAnim_CrReload8 ] {self.frame = 135;} void() PAnim_CrReload8 =[ 9, PAnim_CrReload9 ] {self.frame = 136;} void() PAnim_CrReload9 =[ 10, SUB_Null ] {self.frame = 115;} // Firing, while Crouch void() PAnim_CrouchFire =[ 1, PAnim_CrouchFire1 ] {self.frame = 126; self.tp_anim_time = time + 0.25;} void() PAnim_CrouchFire1=[ 2, SUB_Null ] {self.frame = 127;} // Enter Prone void() PAnim_Prone =[ 1, PAnim_Prone1 ] {self.frame = 154; self.tp_anim_time = time + 1.5;} void() PAnim_Prone1 =[ 2, PAnim_Prone2 ] {self.frame = 155;} void() PAnim_Prone2 =[ 3, PAnim_Prone3 ] {self.frame = 156;} void() PAnim_Prone3 =[ 4, PAnim_Prone4 ] {self.frame = 157;} void() PAnim_Prone4 =[ 5, PAnim_Prone5 ] {self.frame = 158;} void() PAnim_Prone5 =[ 6, PAnim_Prone6 ] {self.frame = 159;} void() PAnim_Prone6 =[ 7, PAnim_Prone7 ] {self.frame = 160;} void() PAnim_Prone7 =[ 8, PAnim_Prone8 ] {self.frame = 161;} void() PAnim_Prone8 =[ 9, SUB_Null ] {self.frame = 162;} // Walking, while Prone void() PAnim_ProneWalk =[ 1, PAnim_ProneWalk1 ] {self.frame = 162; self.tp_anim_time = time + 1.5;} void() PAnim_ProneWalk1 =[ 2, PAnim_ProneWalk2 ] {self.frame = 163;} void() PAnim_ProneWalk2 =[ 3, PAnim_ProneWalk3 ] {self.frame = 164;} void() PAnim_ProneWalk3 =[ 4, PAnim_ProneWalk4 ] {self.frame = 165;} void() PAnim_ProneWalk4 =[ 5, PAnim_ProneWalk5 ] {self.frame = 166;} void() PAnim_ProneWalk5 =[ 6, PAnim_ProneWalk6 ] {self.frame = 167;} void() PAnim_ProneWalk6 =[ 7, PAnim_ProneWalk7 ] {self.frame = 168;} void() PAnim_ProneWalk7 =[ 8, PAnim_ProneWalk8 ] {self.frame = 169;} void() PAnim_ProneWalk8 =[ 9, PAnim_ProneWalk9 ] {self.frame = 170;} void() PAnim_ProneWalk9 =[ 10, PAnim_ProneWalk10 ] {self.frame = 171;} void() PAnim_ProneWalk10=[ 11, PAnim_ProneWalk11 ] {self.frame = 172;} void() PAnim_ProneWalk11=[ 12, PAnim_ProneWalk12 ] {self.frame = 173;} void() PAnim_ProneWalk12=[ 13, SUB_Null ] {self.frame = 174;} // Reloading, while Prone void() PAnim_PrReload =[ 1, PAnim_PrReload1 ] {self.frame = 176; self.tp_anim_time = time + 1;} void() PAnim_PrReload1 =[ 2, PAnim_PrReload2 ] {self.frame = 177;} void() PAnim_PrReload2 =[ 3, PAnim_PrReload3 ] {self.frame = 178;} void() PAnim_PrReload3 =[ 4, PAnim_PrReload4 ] {self.frame = 179;} void() PAnim_PrReload4 =[ 5, PAnim_PrReload5 ] {self.frame = 180;} void() PAnim_PrReload5 =[ 6, PAnim_PrReload6 ] {self.frame = 181;} void() PAnim_PrReload6 =[ 7, SUB_Null ] {self.frame = 162;} // Enter Crouch, from Prone void() PAnim_UpCrouch =[ 1, PAnim_UpCrouch1 ] {self.frame = 161; self.tp_anim_time = time + 1;} void() PAnim_UpCrouch1 =[ 2, PAnim_UpCrouch2 ] {self.frame = 160;} void() PAnim_UpCrouch2 =[ 3, PAnim_UpCrouch3 ] {self.frame = 159;} void() PAnim_UpCrouch3 =[ 4, PAnim_UpCrouch4 ] {self.frame = 158;} void() PAnim_UpCrouch4 =[ 5, PAnim_UpCrouch5 ] {self.frame = 157;} void() PAnim_UpCrouch5 =[ 6, PAnim_UpCrouch6 ] {self.frame = 156;} void() PAnim_UpCrouch6 =[ 7, PAnim_UpCrouch7 ] {self.frame = 155;} void() PAnim_UpCrouch7 =[ 8, PAnim_UpCrouch8 ] {self.frame = 154;} void() PAnim_UpCrouch8 =[ 9, SUB_Null ] {self.frame = 115;} // Enter Last Stand void() PAnim_GetDown =[ 1, PAnim_GetDown1 ] {self.frame = 32; self.tp_anim_time = time + 1;}; void() PAnim_GetDown1 =[ 2, PAnim_GetDown2 ] {self.frame = 33;}; void() PAnim_GetDown2 =[ 3, PAnim_GetDown3 ] {self.frame = 34;}; void() PAnim_GetDown3 =[ 4, PAnim_GetDown4 ] {self.frame = 35;}; void() PAnim_GetDown4 =[ 5, PAnim_GetDown5 ] {self.frame = 36;}; void() PAnim_GetDown5 =[ 6, SUB_Null ] {self.frame = 37;}; // Firing, while in Last Stand void() PAnim_LastFire =[ 1, PAnim_LastFire1 ] {self.frame = 36; self.tp_anim_time = time + 0.25;} void() PAnim_LastFire1 =[ 2, SUB_Null ] {self.frame = 37;} // Leave Last Stand void() PAnim_GetUp =[ 1, PAnim_GetUp1 ] {self.frame = 39; self.tp_anim_time = time + 1;} void() PAnim_GetUp1 =[ 2, PAnim_GetUp2 ] {self.frame = 40;} void() PAnim_GetUp2 =[ 3, PAnim_GetUp3 ] {self.frame = 41;} void() PAnim_GetUp3 =[ 4, PAnim_GetUp4 ] {self.frame = 42;} void() PAnim_GetUp4 =[ 5, PAnim_GetUp5 ] {self.frame = 43;} void() PAnim_GetUp5 =[ 6, PAnim_GetUp6 ] {self.frame = 44;} void() PAnim_GetUp6 =[ 7, PAnim_GetUp7 ] {self.frame = 45;} void() PAnim_GetUp7 =[ 8, PAnim_GetUp8 ] {self.frame = 46;} void() PAnim_GetUp8 =[ 9, PAnim_GetUp9 ] {self.frame = 47;} void() PAnim_GetUp9 =[ 10, SUB_Null ] {self.frame = 48;} #define forward 0 #define backward 1 #define left 2 #define right 3 #define all_move -1 float(float dir) checkMovement = { switch(dir) { case forward: if (self.movement_x > 0) return 1; break; case backward: if (self.movement_x < 0) return 1; break; case right: if (self.movement_y > 0) return 1; break; case left: if (self.movement_y < 0) return 1; break; case all_move: if (self.movement_x || self.movement_y) return 1; break; default: return 0; } } #define Player_RemoveScore(who, value) Player_ChangeScore(who, -value, false) #define Player_AddScore(who, value, impacted_by_2x_points) Player_ChangeScore(who, value, impacted_by_2x_points) // // Player_ChangeScore(who, value, impacted_by_2x_points) // Appends the given value to the provided clients // current score, potentially multiplied if the // Double Points Power-Up is active. // void(entity who, float value, float impacted_by_2x_points) Player_ChangeScore = { // We shouldn't ever allow a player in Last Stand to have // their score modified. if (who.downed) return; // Multiply the value if it is to be impacted by Double // Points if (impacted_by_2x_points && x2_finished > time) value *= 2; // The value was positive, so we should do two things: // 1. add this to our point threshold check for Power-Up // spawning. // 2. add the value to our total score, which is displayed // at the end of the game to rank every player. if (value > 0) { who.score += value; total_powerup_points += value; } // Append the value to our current score. who.points += value; }; // // Player_CanStandHere // Performs a tracebox and will return true // if the player can stance in their current // space. // float(entity who) Player_CanStandHere = { // There shouldn't be any real reason we'd care about this since beta maps don't support // stance changing impacting the bbox, but still, consistency.. if (map_compatibility_mode == MAP_COMPAT_BETA) tracebox(who.origin, PLAYER_MINS_QUAKE, PLAYER_MAXS_QUAKE, who.origin, true, who); else tracebox(who.origin, PLAYER_MINS_STANDING, PLAYER_MAXS_STANDING, who.origin, true, who); return !trace_startsolid; } // // Player_SetStance(who, preferred_stance, play_animation) // Lowers the camera height as needed, registers // preferred stance, plays animation, and sets // bounding box (if applicable). // void(entity who, float preferred_stance, float play_animation) Player_SetStance = { // Don't bother if we're already the desired stance, or if it wasn't valid if (who.stance == preferred_stance || preferred_stance < 0 || preferred_stance > 2) return; // First check -- if we want to stand, only crouch if there // is no space for it. if (preferred_stance == PLAYER_STANCE_STAND && !Player_CanStandHere(who)) preferred_stance = PLAYER_STANCE_CROUCH; // Set the bounding box if (map_compatibility_mode != MAP_COMPAT_BETA) { if (preferred_stance != PLAYER_STANCE_STAND) setsize(self, PLAYER_MINS_CROUCHING, PLAYER_MAXS_CROUCHING); else setsize(self, PLAYER_MINS_STANDING, PLAYER_MAXS_STANDING); } // NZ:P Beta should change the stances by a different amount, // to avoid looking like you're in the ground.. float height_diff_crouch, height_diff_prone; if (map_compatibility_mode == MAP_COMPAT_BETA) { height_diff_crouch = PLAYER_CROUCH_DIFFERENCE_QK; height_diff_prone = PLAYER_PRONE_DIFFERENCE_QK; } else { height_diff_crouch = PLAYER_CROUCH_DIFFERENCE_HL; height_diff_prone = PLAYER_PRONE_DIFFERENCE_HL; } // Prone while standing? Lower to crouch + prone height. if (who.stance == PLAYER_STANCE_STAND && preferred_stance == PLAYER_STANCE_PRONE) who.new_ofs_z = who.view_ofs_z - (height_diff_crouch + height_diff_prone); // Prone while crouching? Lower to prone height. else if (who.stance == PLAYER_STANCE_CROUCH && preferred_stance == PLAYER_STANCE_PRONE) who.new_ofs_z = who.view_ofs_z - height_diff_prone; // Crouch while proning? Raise to crouch height/take away prone difference. else if (who.stance == PLAYER_STANCE_PRONE && preferred_stance == PLAYER_STANCE_CROUCH) who.new_ofs_z = who.view_ofs_z + height_diff_prone; // Crouch while standing? Lower to crouch height. else if (who.stance == PLAYER_STANCE_STAND && preferred_stance == PLAYER_STANCE_CROUCH) who.new_ofs_z = who.view_ofs_z - height_diff_crouch; // Stand while crouching? Raise to stand height/take away crouch difference. else if (who.stance == PLAYER_STANCE_CROUCH && preferred_stance == PLAYER_STANCE_STAND) who.new_ofs_z = who.view_ofs_z + height_diff_crouch; // Stand while proning? Raise to stand height/take away crouch + prone difference. else if (who.stance == PLAYER_STANCE_PRONE && preferred_stance == PLAYER_STANCE_STAND) who.new_ofs_z = who.view_ofs_z + (height_diff_crouch + height_diff_prone); // Set the stance value who.stance = preferred_stance; // Animation playback if (play_animation == true) { entity tempe = self; self = who; switch(who.stance) { case PLAYER_STANCE_STAND: PAnim_Stand(); break; case PLAYER_STANCE_CROUCH: PAnim_Crouch(); break; case PLAYER_STANCE_PRONE: PAnim_Prone(); break; default: break; } self = tempe; } } // // Player_InitiateProgressBar(who, duration) // Kicks-off progress bar for the server to interpret // and render on the client side. // void(entity who, float duration) Player_InitiateProgressBar = { who.progress_bar_time = duration; who.progress_bar = who.progress_bar_time + time; }; // // Player_RemoveProgressBar(who) // Resets progress bar status for client. // void(entity who) Player_RemoveProgressBar = { who.progress_bar = who.progress_bar_time = who.progress_bar_percent = 0; }; // // Player_UpdateProgressBar() // Increments progress bar percentage for client rendering updates. // void() Player_UpdateProgressBar = { if (self.progress_bar_time) { float progress_bar_remaining = self.progress_bar - time; self.progress_bar_percent = invertfloat(progress_bar_remaining / self.progress_bar_time); if (self.progress_bar_percent >= 0.95) Player_RemoveProgressBar(self); } }; void() PlayerJump = { if (!(self.flags & FL_ONGROUND) || !(self.flags & FL_JUMPRELEASED) || self.downed || self.dive ) { return; } self.flags = self.flags - (self.flags & FL_JUMPRELEASED); if (self.button2) self.button2 = 0; self.oldz = self.origin_z; self.velocity_z = 230; } void(float override) JumpCheck = { #ifndef FTE override = 0; #endif // FTE if(self.button2 || override) { if (self.downed) return; if (self.stance == PLAYER_STANCE_STAND) { PlayerJump(); } else if (self.view_ofs_z == self.new_ofs_z && (self.flags & FL_ONGROUND)) { Player_SetStance(self, PLAYER_STANCE_STAND, true); } } else self.flags = self.flags | FL_JUMPRELEASED; } void() PlayerPreThink = { if (self.downed) { self.maxspeed = 30; } else { // Walk slower to account for NZ:P Beta Scaling. if (map_compatibility_mode == MAP_COMPAT_BETA) self.maxspeed = 150; else self.maxspeed = 190; if (self.sprinting) { #ifdef FTE // viewbob when running self.punchangle_y = 0.6*sin(time*10); #endif // FTE //playrun1(); self.maxspeed *= 1.5; // down from 1.66x, to match WaW } else if (!self.sprinting && !self.zoom) { #ifdef FTE if (checkMovement(-1)) self.punchangle_x = 0.25*sin(time*15); #endif // FTE } else if (self.zoom != 3) { self.maxspeed *= 0.5; } // Speed Penalty if (self.speed_penalty_time > time) { self.maxspeed *= self.speed_penalty; } else { self.speed_penalty = 1; } switch(self.stance) { case 1: self.maxspeed *= 0.65; break; case 0: self.maxspeed *= 0.25; break; } #ifdef FTE if (checkMovement(backward)) { self.maxspeed *= 0.7; } else if (checkMovement(left) || checkMovement(right)) { self.maxspeed *= 0.8; } #endif // FTE if (!self.isBuying) self.maxspeed *= GetWeaponWalkSpeed(self.perks, self.weapon); } if(self.isspec != 0 && !self.downed) { if(self.button0 && self.fire_delay < time) { self.aiment = find(self.aiment, classname, "player"); if(self.aiment != world) { sprint(self, PRINT_HIGH, "Now spectating "); sprint(self, PRINT_HIGH, self.aiment.netname); sprint(self, PRINT_HIGH, "\n"); self.solid = SOLID_NOT; self.movetype = MOVETYPE_NONE; } else { sprint(self, PRINT_HIGH, "Freefly spectate\n"); self.movetype = MOVETYPE_FLY; } self.fire_delay = time + 0.25; } if(self.aiment != world) { self.origin = self.aiment.origin; self.angles = self.aiment.v_angle; self.velocity = self.aiment.velocity; self.fixangle = TRUE; } return; } if (cvar("waypoint_mode")) { Waypoint_Logic(); } else { WeaponCore_ClientLogic(); } JumpCheck(0); // Fire effect if (self.onfire) { Effect_Fire(self.origin); if (self.fire_timeout < time) self.onfire = false; } // Flamethrower-based weapon cooldown/refuel process. if (self.ltime < time) { float weapon_slot = Weapon_PlayerHasWeaponWithFiretype(self, FIRETYPE_FLAME); if (weapon_slot == 0) return; weapon_slot--; if (self.weapons[weapon_slot].weapon_magazine == 0 && !self.cooldown) { self.cooldown = true; } if (self.cooldown && self.weapons[weapon_slot].weapon_magazine > 20) self.cooldown = false; if (self.weapons[weapon_slot].weapon_magazine < getWeaponMag(self.weapons[weapon_slot].weapon_id)) { self.weapons[weapon_slot].weapon_magazine += 1; } self.ltime = time + 0.1; } }; float player_trace_time; void() PlayerPostThink = { #ifdef FTE // Network everything self.SendFlags = 1; // Obtain menu state from CSQC via infokeys. self.is_in_menu = stof(infokey(self, "in_menu")); if (self.is_in_menu != 1) if (infokey(self, "chat") == "1") self.is_in_menu = 2; #endif // FTE if(self.isspec) return; //landsound if((self.oldvelocity_z < -212) && (self.flags & FL_ONGROUND)) { Sound_PlaySound(self, "sounds/player/land.wav", SOUND_TYPE_PLAYER_FOOTSTEP, SOUND_PRIORITY_PLAYALWAYS); self.lastsound_time = time - 0.15; } // Walking Animations if (self.tp_anim_time < time && self.reload_delay < time && self.reload_delay2 < time && self.fire_delay < time && self.fire_delay2 < time && self.knife_delay < time) { if (vlen(self.velocity) > 20) { if (self.sprinting) { PAnim_Sprint(); self.tp_anim_type = PLAYER_ANIM_SPRINT; } else { self.tp_anim_type = PLAYER_ANIM_WALK; switch(self.stance) { case 2: PAnim_Walk(); break; case 1: PAnim_CrouchWalk(); break; case 0: PAnim_ProneWalk(); break; } } } else { // Stand still so Crouch Walk isn't stuck in place switch(self.stance) { case 2: self.frame = 0; break; case 1: self.frame = 115; break; case 0: self.frame = 162; break; } // If we're downed force set the last stand idle frame as well // so the player doesn't look like they're just prone. if (self.downed) self.frame = 37; } } #ifdef FTE //footsteps if((vlen(self.velocity) > 20) &&(( time - self.lastsound_time > 0.4) || (time - self.lastsound_time > 0.3 && self.sprinting)) && (self.flags & FL_ONGROUND)) { float movelen = vlen(input_movevalues); if(movelen > 300) { local float ran = random(); if(ran > 0.8) Sound_PlaySound(self, "sounds/player/footstep1.wav", SOUND_TYPE_PLAYER_FOOTSTEP, SOUND_PRIORITY_PLAYALWAYS); else if(ran > 0.6) Sound_PlaySound(self, "sounds/player/footstep2.wav", SOUND_TYPE_PLAYER_FOOTSTEP, SOUND_PRIORITY_PLAYALWAYS); else if(ran > 0.4) Sound_PlaySound(self, "sounds/player/footstep3.wav", SOUND_TYPE_PLAYER_FOOTSTEP, SOUND_PRIORITY_PLAYALWAYS); else if(ran > 0.2) Sound_PlaySound(self, "sounds/player/footstep4.wav", SOUND_TYPE_PLAYER_FOOTSTEP, SOUND_PRIORITY_PLAYALWAYS); else Sound_PlaySound(self, "sounds/player/footstep5.wav", SOUND_TYPE_PLAYER_FOOTSTEP, SOUND_PRIORITY_PLAYALWAYS); self.lastsound_time = time; } } #endif // FTE // Health Regeneration if (self.health_delay < time && self.health != self.max_health && !self.downed) { // If we weren't super low on health, regen is instant. if (self.health_was_very_low == true) { self.health += 120 * frametime; } else { // Was 100% instant, but this delays it by only a few // frames so the graphic doesn't just vanish when healed. self.health += 300 * frametime; } // Make sure we mark that we weren't just super hurt anymore. if (self.max_health <= self.health) { self.health = self.max_health; self.health_was_very_low = false; } } if (self.downed && self.firer != world) { // Has our revivee left the vicinity of our revive trigger? if (fabs(vlen(self.origin - self.firer.origin)) >= 72) { LastStand_UnlinkRevivee(self); } } // Keep track of any active progress bars. Player_UpdateProgressBar(); if (self.sprinting) { self.sprint_timer = self.sprint_duration + (time - self.sprint_start_time); #ifndef FTE if (!self.velocity || self.stance != 2) { W_SprintStop(); } #else if (!self.velocity || !checkMovement(0) || self.stance != 2) { //moto (FIXME) -- move me! W_SprintStop(); } #endif // FTE if (self.perks & P_STAMIN) { if (self.sprint_timer > sprint_max_time * 2) { W_SprintStop(); } } else { if (self.sprint_timer > sprint_max_time) { W_SprintStop(); } } } else if (self.sprint_duration > 0.0) { self.sprint_rest_time = (time - self.sprint_stop_time); } self.oldvelocity = self.velocity; // Perform a traceline to keep track of entities directly // in front of the player. // Also don't do it every frame because that's really tough // on perf. if (player_trace_time < time) { vector source; makevectors (self.v_angle); source = self.origin + self.view_ofs; #ifdef FTE self.dimension_hit = HITBOX_DIM_LIMBS | HITBOX_DIM_ZOMBIES; #endif // FTE traceline (source, source + v_forward*800*1.2, 0, self); #ifdef FTE self.dimension_hit = HITBOX_DIM_ZOMBIES; #endif // FTE // use .head here to avoid expanding ent struct self.head = trace_ent; // check whether we're looking at an entity separately to communicate // with the client more reasonably if (trace_ent.classname == "ai_zombie" || trace_ent.classname == "ai_zombie_head" || trace_ent.classname == "ai_zombie_rarm" ||trace_ent.classname == "ai_zombie_larm" || trace_ent.classname == "ai_dog") self.facingenemy = true; else self.facingenemy = false; #ifndef FTE // 1/4 of a second is enough of a delay to not kill the effect. player_trace_time = time + 0.25; #else // FTE's server rate is already slower, no real need to delay this. player_trace_time = time + 0.05; #endif // FTE } }; void() ClientKill = {}; //called when a client connects to the server void() ClientConnect = { if(cvar("developer") || player_count > 1) { bprint(PRINT_HIGH, self.netname); //print player name bprint(PRINT_HIGH, " connected.\n"); } }; // // Player_PickSpawnPoint() // Picks a valid spawn point for the // newly spawning player, as well as // assigns any spawn-specific fields // (e.g., starting weapon). // void() Player_PickSpawnPoint = { entity spawn_point = world; float found_viable_spawn = false; // Assign a location while(!found_viable_spawn) { float index = random(); // assign one of the spawnpoints if (index < 0.25) spawn_point = find(world, classname, SPAWN_1_CLASS); else if (index < 0.50) spawn_point = find(world, classname, SPAWN_2_CLASS); else if (index < 0.75) spawn_point = find(world, classname, SPAWN_3_CLASS); else spawn_point = find(world, classname, SPAWN_4_CLASS); float found_player_here = false; entity ents_in_spawn_range = findradius(spawn_point.origin, 32); // check if there's a player in the way while(ents_in_spawn_range != world) { if (ents_in_spawn_range.classname == "player") found_player_here = true; ents_in_spawn_range = ents_in_spawn_range.chain; } // no player in the way, this spawn is good. if (found_player_here == false) found_viable_spawn = true; } // Give them their name self.name = spawn_point.name; // Set their location self.origin = spawn_point.origin + '0 0 1'; self.angles = spawn_point.angles; // Assign their starting weapon Weapon_AssignWeapon(0, spawn_point.weapon, spawn_point.currentmag, spawn_point.currentammo); // Activate anything the Spawn was targeting entity tempe = self; self = spawn_point; SUB_UseTargets(); self = tempe; }; void() PlayerSpawn = { entity spawnpoint = world; local_client = self; self.isspec = FALSE; self.classname = "player"; self.solid = SOLID_BBOX; self.flags = FL_CLIENT; self.onfire = false; self.fire_timeout = 0; // We can only collide with zombies (and not their limbs) #ifdef FTE self.dimension_hit = HITBOX_DIM_ZOMBIES; #endif // FTE setmodel(self, "models/player.mdl"); self.movetype = MOVETYPE_WALK; self.max_health = self.health = PLAYER_START_HEALTH; entity who = find(world,classname,"player"); while(who != self && !self.playernum) { if(who) { player_count++; break; } } if (!self.playernum) { self.playernum = player_count + 1; if (self.playernum == 1) pl1 = self; } // Assign them a spawn location. Player_PickSpawnPoint(); self.fixangle = true; // NZ:P Beta used Quake BSP 29, so set bounding box accordingly. if (map_compatibility_mode == MAP_COMPAT_BETA) { self.view_ofs = VIEW_OFS_QK; setsize(self, PLAYER_MINS_QUAKE, PLAYER_MAXS_QUAKE); } else { self.view_ofs = VIEW_OFS_HL; setsize(self, PLAYER_MINS_STANDING, PLAYER_MAXS_STANDING); } self.stance = 2; self.new_ofs_z = self.view_ofs_z; self.oldz = self.origin_z; self.grenades = self.grenades | 1; // add frag grenades to player inventory if (rounds) self.primary_grenades = 2; else self.primary_grenades = 0; // start off without grenades self.pri_grenade_state = 0; // defines that frag grenades are for player first, not betty self.secondary_grenades = -1; // shows that we both don't have betties AND shouldn't draw the image onscreen if (!self.points) Player_AddScore(self, G_STARTPOINTS, false); self.weaponmodel = GetWeaponModel(self.weapon, 0);// Give weapon model self.weapon2model = GetWeapon2Model(self.weapon); self.stamina = 3; self.reviving = 0; self.perks = G_PERKS; UpdatePlayerCount(player_count); #ifdef FTE self.viewzoom = 1; self.SendEntity = Player_SendEntity; self.weapon_animduration = getWeaponDelay(self.weapon, FIRE); if (G_WORLDTEXT) WorldText(world.chaptertitle, world.location, world.date, world.person, self); ReportMapMode(map_compatibility_mode, self); #else self.Weapon_Name = GetWeaponName(self.weapon); self.Flash_Offset = GetWeaponFlash_Offset(self.weapon); self.Flash_Size = GetWeaponFlash_Size(self.weapon); self.ADS_Offset = GetWeaponADSOfs(self.weapon); #endif // FTE if (G_STARTROUND != 1) { rounds = G_STARTROUND - 1; } // Make sure players joining after game start are still allowed // to see their name. if (time >= 5 && self.name != "") nzp_setplayername(self, self.name); }; void() SpectatorSpawn = { local entity spawnpoint; spawnpoint = find(world, classname, "info_player_1_spawn"); self.isspec = TRUE; self.health = 420; self.classname = "spectator"; self.solid = SOLID_CORPSE; setmodel(self, "models/sprites/null.spr"); self.movetype = MOVETYPE_FLY; self.origin = spawnpoint.origin + [0,0,1]; self.fixangle = TRUE; setsize(self, PLAYER_MINS_STANDING, PLAYER_MAXS_STANDING); self.view_ofs = '0 0 22'; self.aiment = world; }; //called when a client loads a map float spawns_initialized; void() PutClientInServer = { // Init Spawn Points if (!spawns_initialized) { Spawns_Init(); spawns_initialized = true; } if(cvar("developer") || player_count > 1) { bprint(PRINT_HIGH, self.netname); bprint(PRINT_HIGH, " has joined the game.\n"); } if (spawn_time > time || !rounds) PlayerSpawn(); #ifdef FTE else SpectatorSpawn(); // Force the client to always be networked to other clients, even when // outside of the same PVS. self.pvsflags |= PVSF_IGNOREPVS; #endif // FTE GameRestart_ResetPerkaColas(); }; //called when client disconnects from the server void() ClientDisconnect = { bprint(PRINT_HIGH, self.netname); bprint(PRINT_HIGH, " has left the game.\n"); player_count--; UpdatePlayerCount(player_count); self.classname = "disconnected"; self.solid = SOLID_NOT; self.movetype = MOVETYPE_BOUNCE; self.nextthink = -1; setmodel(self, "models/sprites/null.spr"); GameRestart_ResetPerkaColas(); if (self.downed && self.firer != world) { LastStand_UnlinkRevivee(self); DisableReviveIcon(self.playernum); } else if (self.firer != world) { self.firer.beingrevived = false; } #ifdef FTE // Network everything self.SendFlags = 1; self.is_in_menu = 0; #endif // FTE }; void() SetNewParms = { }; void() SetChangeParms = { }; #ifdef FTE void() SV_RunClientCommand = { runstandardplayerphysics(self); } #endif // FTE