From 0bf23224a4d38f9a370569ae9083d1b7fa960c5b Mon Sep 17 00:00:00 2001 From: Mario Date: Thu, 8 May 2014 03:40:42 +1000 Subject: [PATCH] Add an option to always show 3rd person view in nexball while not carrying weapons, show a waypoint at the enemy's goal so players know where to take the ball, fix nexball showing up as a modification rather than a gamemode --- defaultXonotic.cfg | 1 + qcsrc/client/View.qc | 3 +- qcsrc/client/autocvars.qh | 1 + qcsrc/client/waypointsprites.qc | 1 + qcsrc/server/cl_physics.qc | 4 +- qcsrc/server/mutators/gamemode_nexball.qc | 94 ++++++++++++++++------- 6 files changed, 73 insertions(+), 31 deletions(-) diff --git a/defaultXonotic.cfg b/defaultXonotic.cfg index 747e0c419..e549ca62a 100644 --- a/defaultXonotic.cfg +++ b/defaultXonotic.cfg @@ -224,6 +224,7 @@ seta cl_hitsound 1 "play a hit notifier sound when you have hit an enemy" set cl_hitsound_antispam_time 0.05 "don't play the hitsound more often than this" seta cl_eventchase_death 1 "camera goes into 3rd person mode when the player is dead" +seta cl_eventchase_nexball 1 "camera goes into 3rd person mode when in nexball game-mode" seta cl_eventchase_distance 140 "final camera distance" seta cl_eventchase_speed 1.3 "how fast the camera slides back, 0 is instant" seta cl_eventchase_maxs "12 12 8" "max size of eventchase camera bbox" diff --git a/qcsrc/client/View.qc b/qcsrc/client/View.qc index 85dfe547c..2a41ba13c 100644 --- a/qcsrc/client/View.qc +++ b/qcsrc/client/View.qc @@ -492,7 +492,8 @@ void CSQC_UpdateView(float w, float h) // event chase camera if(autocvar_chase_active <= 0) // greater than 0 means it's enabled manually, and this code is skipped { - if(((spectatee_status >= 0 && (autocvar_cl_eventchase_death && is_dead)) || intermission) && !autocvar_cl_orthoview) + WepSet weapons_stat = WepSet_GetFromStat(); + if(((spectatee_status >= 0 && (autocvar_cl_eventchase_death && is_dead)) || intermission) && !autocvar_cl_orthoview || (autocvar_cl_eventchase_nexball && gametype == MAPINFO_TYPE_NEXBALL && !(weapons_stat & WepSet_FromWeapon(WEP_PORTO)))) { // make special vector since we can't use view_origin (It is one frame old as of this code, it gets set later with the results this code makes.) vector current_view_origin = (csqcplayer ? csqcplayer.origin : pmove_org); diff --git a/qcsrc/client/autocvars.qh b/qcsrc/client/autocvars.qh index 5ba321ae1..67a032593 100644 --- a/qcsrc/client/autocvars.qh +++ b/qcsrc/client/autocvars.qh @@ -410,6 +410,7 @@ float autocvar_viewsize; float autocvar_cl_hitsound; float autocvar_cl_hitsound_antispam_time; var float autocvar_cl_eventchase_death = 1; +var float autocvar_cl_eventchase_nexball = 1; var float autocvar_cl_eventchase_distance = 140; var float autocvar_cl_eventchase_speed = 1.3; var vector autocvar_cl_eventchase_maxs = '12 12 8'; diff --git a/qcsrc/client/waypointsprites.qc b/qcsrc/client/waypointsprites.qc index dd7ae36b9..1367501a8 100644 --- a/qcsrc/client/waypointsprites.qc +++ b/qcsrc/client/waypointsprites.qc @@ -277,6 +277,7 @@ string spritelookuptext(string s) case "race-finish": return _("Finish"); case "race-start": return _("Start"); case "race-start-finish": return (race_checkpointtime || race_mycheckpointtime) ? _("Finish") : _("Start"); + case "goal": return _("Goal"); case "nb-ball": return _("Ball"); case "ka-ball": return _("Ball"); case "ka-ballcarrier": return _("Ball carrier"); diff --git a/qcsrc/server/cl_physics.qc b/qcsrc/server/cl_physics.qc index 009856137..0d0dd3138 100644 --- a/qcsrc/server/cl_physics.qc +++ b/qcsrc/server/cl_physics.qc @@ -636,9 +636,7 @@ void SV_PlayerPhysics() maxspd_mod = 1; if(self.ballcarried) - if(g_nexball) - maxspd_mod *= autocvar_g_nexball_basketball_carrier_highspeed; - else if(g_keepaway) + if(g_keepaway) maxspd_mod *= autocvar_g_keepaway_ballcarrier_highspeed; maxspd_mod *= autocvar_g_movement_highspeed; diff --git a/qcsrc/server/mutators/gamemode_nexball.qc b/qcsrc/server/mutators/gamemode_nexball.qc index 2d2c857e9..014d37ec2 100644 --- a/qcsrc/server/mutators/gamemode_nexball.qc +++ b/qcsrc/server/mutators/gamemode_nexball.qc @@ -457,11 +457,21 @@ void nb_spawnteams(void) } } +// scoreboard setup +void nb_ScoreRules(float teams) +{ + ScoreRules_basics(teams, 0, 0, TRUE); + ScoreInfo_SetLabel_TeamScore( ST_NEXBALL_GOALS, "goals", SFL_SORT_PRIO_PRIMARY); + ScoreInfo_SetLabel_PlayerScore( SP_NEXBALL_GOALS, "goals", SFL_SORT_PRIO_PRIMARY); + ScoreInfo_SetLabel_PlayerScore(SP_NEXBALL_FAULTS, "faults", SFL_SORT_PRIO_SECONDARY | SFL_LOWER_IS_BETTER); + ScoreRules_basics_end(); +} + void nb_delayedinit(void) { if(find(world, classname, "nexball_team") == world) nb_spawnteams(); - ScoreRules_nexball(nb_teams); + nb_ScoreRules(nb_teams); } @@ -471,11 +481,7 @@ void nb_delayedinit(void) void SpawnBall(void) { - if(!g_nexball) - { - remove(self); - return; - } + if(!g_nexball) { remove(self); return; } // balls += 4; // using the remaining bits to count balls will leave more than the max edict count, so it's fine @@ -559,14 +565,28 @@ void spawnfunc_nexball_football(void) SpawnBall(); } +float nb_Goal_Customize() +{ + entity e, wp_owner; + e = WaypointSprite_getviewentity(other); + wp_owner = self.owner; + if(SAME_TEAM(e, wp_owner)) { return FALSE; } + + return TRUE; +} + void SpawnGoal(void) { - if(!g_nexball) + if(!g_nexball) { remove(self); return; } + + EXACTTRIGGER_INIT; + + if(self.team != GOAL_OUT && Team_TeamToNumber(self.team) != -1) { - remove(self); - return; + WaypointSprite_SpawnFixed("goal", (self.absmin + self.absmax) * 0.5, self, sprite, RADARICON_NONE, ((self.team) ? Team_ColorRGB(self.team) : '1 0.5 0')); + self.sprite.customizeentityforclient = nb_Goal_Customize; } - EXACTTRIGGER_INIT; + self.classname = "nexball_goal"; if(self.noise == "") self.noise = "ctf/respawn.wav"; @@ -674,7 +694,7 @@ void W_Nexball_Touch(void) PROJECTILE_TOUCH; if(attacker.team != other.team || autocvar_g_nexball_basketball_teamsteal) - if((ball = other.ballcarried) && (IS_PLAYER(attacker))) + if((ball = other.ballcarried) && !other.deadflag && (IS_PLAYER(attacker))) { other.velocity = other.velocity + normalize(self.velocity) * other.damageforcescale * autocvar_g_balance_nexball_secondary_force; other.flags &= ~FL_ONGROUND; @@ -683,7 +703,7 @@ void W_Nexball_Touch(void) LogNB("stole", attacker); sound(other, CH_TRIGGER, ball.noise2, VOL_BASE, ATTEN_NORM); - if(attacker.team == other.team && time > attacker.teamkill_complain) + if(SAME_TEAM(attacker, other) && time > attacker.teamkill_complain) { attacker.teamkill_complain = time + 5; attacker.teamkill_soundtime = time + 0.4; @@ -760,7 +780,7 @@ void W_Nexball_Attack2(void) missile.movetype = MOVETYPE_FLY; PROJECTILE_MAKETRIGGER(missile); - setmodel(missile, "models/elaser.mdl"); // precision set below + //setmodel(missile, "models/elaser.mdl"); // precision set below setsize(missile, '0 0 0', '0 0 0'); setorigin(missile, w_shotorg); @@ -772,6 +792,8 @@ void W_Nexball_Attack2(void) missile.effects = EF_BRIGHTFIELD | EF_LOWPRECISION; missile.flags = FL_PROJECTILE; + + CSQCProjectile(missile, TRUE, PROJECTILE_ELECTRO, TRUE); } float ball_customize() @@ -859,18 +881,6 @@ MUTATOR_HOOKFUNCTION(nexball_BallDrop) return 0; } -MUTATOR_HOOKFUNCTION(nexball_BuildMutatorsString) -{ - ret_string = strcat(ret_string, ":NB"); - return 0; -} - -MUTATOR_HOOKFUNCTION(nexball_BuildMutatorsPrettyString) -{ - ret_string = strcat(ret_string, ", NexBall"); - return 0; -} - MUTATOR_HOOKFUNCTION(nexball_PlayerPreThink) { makevectors(self.v_angle); @@ -947,6 +957,16 @@ MUTATOR_HOOKFUNCTION(nexball_PlayerSpawn) return FALSE; } +MUTATOR_HOOKFUNCTION(nexball_PlayerPhysics) +{ + if(self.ballcarried) + { + self.stat_sv_airspeedlimit_nonqw *= autocvar_g_nexball_basketball_carrier_highspeed; + self.stat_sv_maxspeed *= autocvar_g_nexball_basketball_carrier_highspeed; + } + return FALSE; +} + MUTATOR_HOOKFUNCTION(nexball_SetStartItems) { start_items |= IT_UNLIMITED_SUPERWEAPONS; // FIXME BAD BAD BAD BAD HACK, NEXBALL SHOULDN'T ABUSE PORTO'S WEAPON SLOT @@ -954,16 +974,34 @@ MUTATOR_HOOKFUNCTION(nexball_SetStartItems) return FALSE; } +MUTATOR_HOOKFUNCTION(nexball_ForbidThrowing) +{ + if(self.weapon == WEP_GRENADE_LAUNCHER) + return TRUE; + + return FALSE; +} + +MUTATOR_HOOKFUNCTION(nexball_FilterItem) +{ + if(self.classname == "droppedweapon") + if(self.weapon == WEP_GRENADE_LAUNCHER) + return TRUE; + + return FALSE; +} + MUTATOR_DEFINITION(gamemode_nexball) { MUTATOR_HOOK(PlayerDies, nexball_BallDrop, CBC_ORDER_ANY); MUTATOR_HOOK(MakePlayerObserver, nexball_BallDrop, CBC_ORDER_ANY); MUTATOR_HOOK(ClientDisconnect, nexball_BallDrop, CBC_ORDER_ANY); - MUTATOR_HOOK(BuildMutatorsPrettyString, nexball_BuildMutatorsPrettyString, CBC_ORDER_ANY); - MUTATOR_HOOK(BuildMutatorsString, nexball_BuildMutatorsString, CBC_ORDER_ANY); MUTATOR_HOOK(PlayerSpawn, nexball_PlayerSpawn, CBC_ORDER_ANY); MUTATOR_HOOK(PlayerPreThink, nexball_PlayerPreThink, CBC_ORDER_ANY); + MUTATOR_HOOK(PlayerPhysics, nexball_PlayerPhysics, CBC_ORDER_ANY); MUTATOR_HOOK(SetStartItems, nexball_SetStartItems, CBC_ORDER_ANY); + MUTATOR_HOOK(ForbidThrowCurrentWeapon, nexball_ForbidThrowing, CBC_ORDER_ANY); + MUTATOR_HOOK(FilterItem, nexball_FilterItem, CBC_ORDER_ANY); MUTATOR_ONADD { @@ -973,6 +1011,8 @@ MUTATOR_DEFINITION(gamemode_nexball) g_nexball_meter_period = rint(g_nexball_meter_period * 32) / 32; //Round to 1/32ths to send as a byte multiplied by 32 addstat(STAT_NB_METERSTART, AS_FLOAT, metertime); + w_porto(WR_PRECACHE); // abuse + // General settings /* CVTOV(g_nexball_football_boost_forward); //100 -- 2.39.2