From: Mario Date: Mon, 7 Oct 2013 23:44:33 +0000 (-0700) Subject: Merge branch 'Mario/spawnsystem_mutator' into Mario/overkill X-Git-Tag: xonotic-v0.8.0~70^2~27 X-Git-Url: https://git.rm.cloudns.org/?a=commitdiff_plain;h=77477d386d12b458eaa8dabb23853ef2e391a71a;p=xonotic%2Fxonotic-data.pk3dir.git Merge branch 'Mario/spawnsystem_mutator' into Mario/overkill --- 77477d386d12b458eaa8dabb23853ef2e391a71a diff --cc mutators.cfg index b25fb138fa,c762ef9dec..ef60d81934 --- a/mutators.cfg +++ b/mutators.cfg @@@ -33,39 -33,6 +33,34 @@@ set g_minstagib_invis_alpha 0.1 set g_minstagib_speed_highspeed 1.5 "speed-multiplier that applies while you carry the invincibility powerup" +// ========== +// overkill +// ========== +set g_overkill 0 "enable overkill" - set g_overkill_spawnsystem 1 "experimental spawnsystem" - set g_overkill_spawnsystem_delay 2.5 "how long to wait before its OK to spawn at a player after someone just spawned at this player" - set g_overkill_spawnsystem_delay_death 0 "how long to wait before its OK to spawn at a player after death " - set g_overkill_spawnsystem_check_health 1 "only allow spawn at this player if hes health is full" - set g_overkill_spawnsystem_close2death 1 "spawn as close as possible to where we died" + +set g_overkill_100a_anyway 1 +set g_overkill_100h_anyway 1 +set g_overkill_powerups_replace 1 +set g_overkill_superguns_onemag 1 +set g_overkill_superguns_respawn_time 20 + +set g_overkill_ammo_charge 1 +set g_overkill_ammo_charge_notice 1 +set g_overkill_ammo_charge_limit 1 +set g_overkill_ammo_charge_rate 0.5 +set g_overkill_ammo_charge_rate_nex 0.5 +set g_overkill_ammo_charge_rate_uzi 0.5 +set g_overkill_ammo_charge_rate_shotgun 0.5 +set g_overkill_ammo_charge_rate_hmg 0.25 +set g_overkill_ammo_charge_rate_rpc 1.5 +set g_overkill_ammo_decharge 0.1 +set g_overkill_ammo_decharge_uzi 0.025 +set g_overkill_ammo_decharge_shotgun 0.15 +set g_overkill_ammo_decharge_nex 0.2 +set g_overkill_ammo_decharge_rpc 1 +set g_overkill_ammo_decharge_hmg 0.01 + + // ========= // vampire // ========= diff --cc qcsrc/server/autocvars.qh index 808186c60c,299759dd2c..1909ee499f --- a/qcsrc/server/autocvars.qh +++ b/qcsrc/server/autocvars.qh @@@ -1280,16 -1254,9 +1280,17 @@@ float autocvar_g_campcheck_damage float autocvar_g_campcheck_distance; float autocvar_g_campcheck_interval; float autocvar_g_jump_grunt; - float autocvar_g_overkill_spawnsystem; - float autocvar_g_overkill_spawnsystem_delay; - float autocvar_g_overkill_spawnsystem_delay_death; - float autocvar_g_overkill_spawnsystem_check_health; - float autocvar_g_overkill_spawnsystem_close2death; +float autocvar_g_overkill_powerups_replace; +float autocvar_g_overkill_superguns_respawn_time; +float autocvar_g_overkill_100h_anyway; +float autocvar_g_overkill_100a_anyway; +float autocvar_g_overkill_ammo_start; +float autocvar_g_overkill_ammo_charge; +float autocvar_g_overkill_ammo_charge_notice; +float autocvar_g_overkill_ammo_charge_limit; + float autocvar_g_spawn_near_teammate_distance; + float autocvar_g_spawn_near_teammate_ignore_spawnpoint; + float autocvar_g_spawn_near_teammate_ignore_spawnpoint_delay; + float autocvar_g_spawn_near_teammate_ignore_spawnpoint_delay_death; + float autocvar_g_spawn_near_teammate_ignore_spawnpoint_check_health; + float autocvar_g_spawn_near_teammate_ignore_spawnpoint_closetodeath; diff --cc qcsrc/server/mutators/mutator_overkill.qc index 228a1f1523,0000000000..713866d97a mode 100644,000000..100644 --- a/qcsrc/server/mutators/mutator_overkill.qc +++ b/qcsrc/server/mutators/mutator_overkill.qc @@@ -1,466 -1,0 +1,378 @@@ +void ok_DecreaseCharge(entity ent, float wep) +{ + if not(autocvar_g_overkill_ammo_charge) return; + + entity wepent = get_weaponinfo(wep); + + if(wepent.netname == "") + return; // dummy + + ent.ammo_charge[wep] -= cvar(sprintf("g_overkill_ammo_decharge_%s", wepent.netname)); +} + +void ok_IncreaseCharge(entity ent, float wep) +{ + entity wepent = get_weaponinfo(wep); + + if(wepent.netname == "") + return; // dummy + + if(autocvar_g_overkill_ammo_charge) + if not(ent.BUTTON_ATCK) + if(ent.ammo_charge[wep] <= autocvar_g_overkill_ammo_charge_limit) + ent.ammo_charge[wep] = min(autocvar_g_overkill_ammo_charge_limit, ent.ammo_charge[wep] + cvar(sprintf("g_overkill_ammo_charge_rate_%s", wepent.netname)) * frametime / W_TICSPERFRAME); +} + +float ok_CheckWeaponCharge(entity ent, float wep, float swep) +{ + if not(autocvar_g_overkill_ammo_charge) return TRUE; + //if(wep != swep) return TRUE; // switching, don't even bother checking + + entity wepent = get_weaponinfo(wep); + + if(wepent.netname == "") + return 0; // dummy + + return !(ent.ammo_charge[wep] < cvar(sprintf("g_overkill_ammo_decharge_%s", wepent.netname))); +} + +MUTATOR_HOOKFUNCTION(ok_PlayerDamage_Calculate) +{ + if(DEATH_ISWEAPON(frag_deathtype, WEP_LASER) && frag_target == frag_attacker) + frag_damage = 0; + + if(!IS_PLAYER(frag_target) || frag_target == frag_attacker) + return FALSE; + + if not (DEATH_ISWEAPON(frag_deathtype, WEP_LASER)) + return FALSE; + + frag_damage = 0; + if(frag_target.health >= 1 && !frag_target.freezetag_frozen) + Send_Notification(NOTIF_ONE, frag_attacker, MSG_CENTER, CENTER_MINSTA_SECONDARY); + + frag_force = '0 0 0'; + + return FALSE; +} + +MUTATOR_HOOKFUNCTION(ok_BuildMutatorsString) +{ + ret_string = strcat(ret_string, ":OK"); + return FALSE; +} + +MUTATOR_HOOKFUNCTION(ok_BuildMutatorsPrettyString) +{ + ret_string = strcat(ret_string, ", Overkill"); + return FALSE; +} + +void ok_Item_Touch() +{ + if(IS_PLAYER(other) && other.deadflag == DEAD_NO && !other.freezetag_frozen) + { + Item_Touch(); + remove(self); + } +} + +MUTATOR_HOOKFUNCTION(ok_PlayerDies) +{ + self.ok_lastwep = self.weapon; + self.ok_deathloc = self.origin; + + entity oldself = self; + float gpi = autocvar_g_pickup_items; + + cvar_set("g_pickup_items", "1"); + self = spawn(); + self.ok_item = TRUE; + self.noalign = TRUE; + spawnfunc_item_armor_small(); + self.movetype = MOVETYPE_TOSS; + self.gravity = 1; + setorigin(self, frag_target.origin + '0 0 32'); + self.velocity = '0 0 200' + normalize(frag_attacker.origin - self.origin) * 500; + self.touch = ok_Item_Touch; + SUB_SetFade(self, time + 5, 1); + self = oldself; + cvar_set("g_pickup_items", ftos(gpi)); + + return FALSE; +} + +MUTATOR_HOOKFUNCTION(ok_ForbidThrowCurrentWeapon) +{ + nades_CheckThrow(); + + return TRUE; +} + +MUTATOR_HOOKFUNCTION(ok_PlayerPreThink) +{ + if(intermission_running || gameover) + return FALSE; + + if(self.deadflag != DEAD_NO || !IS_PLAYER(self)) + return FALSE; + + if(self.freezetag_frozen) + return FALSE; + + ok_IncreaseCharge(self, self.weapon); + + if(self.BUTTON_ATCK2) + if(self.jump_interval <= time) + { + self.jump_interval = time + autocvar_g_balance_laser_primary_refire * W_WeaponRateFactor(); + makevectors(self.v_angle); + float w = self.weapon; + self.weapon = WEP_LASER; + W_Laser_Attack(3); + self.weapon = w; + } + + self.weapon_blocked = FALSE; + + self.ok_ammo_charge = self.ammo_charge[self.weapon]; + + if(autocvar_g_overkill_ammo_charge) + if(!ok_CheckWeaponCharge(self, self.weapon, self.switchweapon)) + { + if(autocvar_g_overkill_ammo_charge_notice && time > self.ok_notice_time && self.BUTTON_ATCK && IS_REAL_CLIENT(self)) + { + //Send_Notification(NOTIF_ONE, self, MSG_CENTER, CENTER_OVERKILL_CHARGE); + self.ok_notice_time = time + 2; + play2(self, "weapons/dryfire.wav"); + } + if(self.weaponentity.state != WS_CLEAR) + w_ready(); + + self.weapon_blocked = TRUE; + } + + self.BUTTON_ATCK2 = 0; + + return FALSE; +} + +MUTATOR_HOOKFUNCTION(ok_PlayerSpawn) +{ + if(autocvar_g_overkill_ammo_charge) + { + float i; + + for(i = WEP_FIRST; i <= WEP_LAST; ++i) + self.ammo_charge[i] = autocvar_g_overkill_ammo_charge_limit; + + self.ok_use_ammocharge = 1; + self.ok_notice_time = time; + } + else + self.ok_use_ammocharge = 0; + - if(autocvar_g_overkill_spawnsystem_delay_death) - self.ok_spawnsys_timer = time + autocvar_g_overkill_spawnsystem_delay_death; - - if(teamplay == 0 || autocvar_g_overkill_spawnsystem == 0) - return FALSE; - - entity team_mate, best_mate = world; - vector best_spot = '0 0 0'; - float pc = 0., best_dist = 0., dist = 0.; - FOR_EACH_PLAYER(team_mate) - { - if((autocvar_g_overkill_spawnsystem_check_health != 0 && team_mate.health >= 100) || autocvar_g_overkill_spawnsystem_check_health == 0) - if(team_mate.deadflag == DEAD_NO) - if(team_mate.ok_spawnsys_timer < time) - if(team_mate.team == self.team && (g_ctf || g_tdm || g_keyhunt)) - if(team_mate != self) - { - tracebox(team_mate.origin, PL_MIN, PL_MAX, team_mate.origin - '0 0 100', MOVE_WORLDONLY, team_mate); - if(trace_fraction != 1.0) - if not(trace_dphitq3surfaceflags & Q3SURFACEFLAG_SKY) - { - pc = pointcontents(trace_endpos + '0 0 1'); - if(pc == CONTENT_EMPTY) - { - if(vlen(team_mate.velocity) > 5) - fixedmakevectors(vectoangles(team_mate.velocity)); - else - fixedmakevectors(team_mate.angles); - - for(pc = 0; pc != 5; ++pc) // test 5 diffrent spots close to mate - { - switch(pc) - { - case 0: - tracebox(team_mate.origin , PL_MIN, PL_MAX, team_mate.origin + v_right * 128, MOVE_NORMAL, team_mate); - break; - case 1: - tracebox(team_mate.origin , PL_MIN, PL_MAX,team_mate.origin - v_right * 128 , MOVE_NORMAL, team_mate); - break; - case 2: - tracebox(team_mate.origin , PL_MIN, PL_MAX, team_mate.origin + v_right * 64 - v_forward * 64, MOVE_NORMAL, team_mate); - break; - case 3: - tracebox(team_mate.origin , PL_MIN, PL_MAX, team_mate.origin - v_right * 64 - v_forward * 64, MOVE_NORMAL, team_mate); - break; - case 4: - tracebox(team_mate.origin , PL_MIN, PL_MAX, team_mate.origin - v_forward * 128, MOVE_NORMAL, team_mate); - break; - } - - if(trace_fraction == 1.0) - { - traceline(trace_endpos + '0 0 4', trace_endpos - '0 0 100', MOVE_NORMAL, team_mate); - if(trace_fraction != 1.0) - { - if(autocvar_g_overkill_spawnsystem_close2death) - { - dist = vlen(trace_endpos - self.ok_deathloc); - if(dist < best_dist || best_dist == 0) - { - best_dist = dist; - best_spot = trace_endpos; - best_mate = team_mate; - } - } - else - { - setorigin(self, trace_endpos); - self.angles = team_mate.angles; - team_mate.ok_spawnsys_timer = time + autocvar_g_overkill_spawnsystem_delay; - return FALSE; - } - } - } - } - } - } - } - } - - if(autocvar_g_overkill_spawnsystem_close2death) - if(best_dist) - { - setorigin(self, best_spot); - self.angles = best_mate.angles; - best_mate.ok_spawnsys_timer = time + autocvar_g_overkill_spawnsystem_delay; - } - - return 0; ++ return FALSE; +} + +void start_hmg() +{ + float gpi = autocvar_g_pickup_items; + cvar_set("g_pickup_items", "1"); + self.classname = "weapon_hmg"; + self.ok_item = TRUE; + //self.weapons = WEP_HMG; + //self.flags |= FL_POWERUP; + self.respawntime = autocvar_g_overkill_superguns_respawn_time; + setmodel(self, "models/weapons/g_ok_hmg.md3"); + spawnfunc_weapon_hmg(); + + cvar_set("g_pickup_items", ftos(gpi)); +} + +void start_rpc() +{ + float gpi = autocvar_g_pickup_items; + cvar_set("g_pickup_items", "1"); + self.classname = "weapon_rpc"; + self.respawntime = autocvar_g_overkill_superguns_respawn_time; + self.ok_item = TRUE; + //self.weapons = WEP_RPC; + //self.flags |= FL_POWERUP; + setmodel(self, "models/weapons/g_ok_rl.md3"); + spawnfunc_weapon_rpc(); + cvar_set("g_pickup_items", ftos(gpi)); +} + +void start_mh_anyway() +{ + cvar_set("g_pickup_items", "1"); + self.classname = "item_health_mega"; + self.ok_item = TRUE; + spawnfunc_item_health_mega(); + cvar_set("g_pickup_items", "0"); +} + +void start_ma_anyway() +{ + cvar_set("g_pickup_items", "1"); + self.classname = "item_armor_large"; + self.ok_item = TRUE; + spawnfunc_item_armor_large(); + cvar_set("g_pickup_items", "0"); +} + +MUTATOR_HOOKFUNCTION(ok_OnEntityPreSpawn) +{ + if(autocvar_g_powerups) + if(autocvar_g_overkill_powerups_replace) + if(self.classname == "item_strength") + { + entity wep = spawn(); + setorigin(wep, self.origin); + wep.ok_item = TRUE; + wep.think = start_hmg; + wep.nextthink = time + 0.1; + return TRUE; + } + + if(autocvar_g_powerups) + if(autocvar_g_overkill_powerups_replace) + if(self.classname == "item_invincible") + { + entity wep = spawn(); + setorigin(wep, self.origin); + wep.ok_item = TRUE; + wep.think = start_rpc; + wep.nextthink = time + 0.1; + return TRUE; + } + + return FALSE; +} + +MUTATOR_HOOKFUNCTION(ok_ItemRemove) +{ + if(self.ok_item) + return FALSE; + + switch(self.items) + { + case IT_HEALTH: return !(autocvar_g_overkill_100h_anyway); + case IT_ARMOR: return !(autocvar_g_overkill_100a_anyway); + } + + return TRUE; +} + +MUTATOR_HOOKFUNCTION(ok_SpectateCopy) +{ + self.ammo_charge[self.weapon] = other.ammo_charge[other.weapon]; + self.ok_use_ammocharge = other.ok_use_ammocharge; + + return FALSE; +} + +MUTATOR_HOOKFUNCTION(ok_SetModname) +{ + modname = "Overkill"; + return TRUE; +} + +MUTATOR_HOOKFUNCTION(ok_StartItems) +{ + start_items |= IT_UNLIMITED_WEAPON_AMMO; + start_weapons = (WEPSET_UZI | WEPSET_NEX | WEPSET_SHOTGUN); + + start_ammo_nails = start_ammo_cells = start_ammo_shells = start_ammo_rockets = + warmup_start_ammo_nails = warmup_start_ammo_cells = warmup_start_ammo_shells = warmup_start_ammo_rockets = autocvar_g_overkill_ammo_start; + + return FALSE; +} + +void ok_Initialize() +{ + precache_all_playermodels("models/ok_player/*.dpm"); + + precache_model("models/weapons/h_ok_rl.iqm"); + precache_model("models/weapons/v_ok_rl.md3"); + precache_model("models/weapons/g_ok_rl.md3"); + precache_model("models/weapons/ok_rocket.md3"); + + precache_model("models/weapons/h_ok_mg.iqm"); + precache_model("models/weapons/v_ok_mg.md3"); + precache_model("models/weapons/g_ok_mg.md3"); + + precache_model("models/weapons/h_ok_hmg.iqm"); + precache_model("models/weapons/v_ok_hmg.md3"); + precache_model("models/weapons/g_ok_hmg.md3"); + + precache_model("models/weapons/h_ok_shotgun.iqm"); + precache_model("models/weapons/v_ok_shotgun.md3"); + precache_model("models/weapons/g_ok_shotgun.md3"); + + precache_model("models/weapons/h_ok_sniper.iqm"); + precache_model("models/weapons/v_ok_sniper.md3"); + precache_model("models/weapons/g_ok_sniper.md3"); + + precache_sound("weapons/dryfire.wav"); + + addstat(STAT_OK_AMMO_CHARGE, AS_FLOAT, ok_use_ammocharge); + addstat(STAT_OK_AMMO_CHARGEPOOl, AS_FLOAT, ok_ammo_charge); + + w_uzi(WR_PRECACHE); + w_nex(WR_PRECACHE); + w_shotgun(WR_PRECACHE); + w_laser(WR_PRECACHE); + + (get_weaponinfo(WEP_RPC)).spawnflags &= ~WEP_FLAG_MUTATORBLOCKED; + (get_weaponinfo(WEP_HMG)).spawnflags &= ~WEP_FLAG_MUTATORBLOCKED; + + (get_weaponinfo(WEP_SHOTGUN)).mdl = "ok_shotgun"; + (get_weaponinfo(WEP_UZI)).mdl = "ok_mg"; + (get_weaponinfo(WEP_NEX)).mdl = "ok_sniper"; + + string s; + float fh = fopen("overkill.cfg", FILE_READ); + if(fh >= 0) + { + while((s = fgets(fh))) + { + tokenize_console(s); + if not(argv(0) == "" || argv(1) == "//" || argv(1) == "") + cvar_settemp(argv(0), argv(1)); + } + fclose(fh); + } + else + dprint("^1Mutator Overkill: WARNING! overkill.cfg NOT found, things will be strange!\n"); +} + +MUTATOR_DEFINITION(mutator_overkill) +{ + MUTATOR_HOOK(ForbidThrowCurrentWeapon, ok_ForbidThrowCurrentWeapon, CBC_ORDER_ANY); + MUTATOR_HOOK(BuildMutatorsString, ok_BuildMutatorsString, CBC_ORDER_ANY); + MUTATOR_HOOK(BuildMutatorsPrettyString, ok_BuildMutatorsPrettyString, CBC_ORDER_ANY); + MUTATOR_HOOK(PlayerPreThink, ok_PlayerPreThink, CBC_ORDER_LAST); + MUTATOR_HOOK(PlayerSpawn, ok_PlayerSpawn, CBC_ORDER_LAST); + MUTATOR_HOOK(PlayerDamage_Calculate, ok_PlayerDamage_Calculate, CBC_ORDER_LAST); + MUTATOR_HOOK(PlayerDies, ok_PlayerDies, CBC_ORDER_ANY); + MUTATOR_HOOK(OnEntityPreSpawn, ok_OnEntityPreSpawn, CBC_ORDER_ANY); + MUTATOR_HOOK(SetModname, ok_SetModname, CBC_ORDER_ANY); + MUTATOR_HOOK(FilterItem, ok_ItemRemove, CBC_ORDER_ANY); + MUTATOR_HOOK(SpectateCopy, ok_SpectateCopy, CBC_ORDER_ANY); + MUTATOR_HOOK(SetStartItems, ok_StartItems, CBC_ORDER_ANY); + + MUTATOR_ONADD + { + ok_Initialize(); + } + + MUTATOR_ONREMOVE + { + (get_weaponinfo(WEP_RPC)).spawnflags |= WEP_FLAG_MUTATORBLOCKED; + (get_weaponinfo(WEP_HMG)).spawnflags |= WEP_FLAG_MUTATORBLOCKED; + } + + return FALSE; +}