prozac-qfcc/tfortmap.qc
Finny Merrill 043ddfa55c Continuing to merge in changes between 2.5 - 2.9
Added support for all the new toggleflags, full team score, war standard and
flag emulation.
2004-02-01 12:06:03 +00:00

3669 lines
85 KiB
C++
Raw Blame History

#include "defs.qh"
#include "messages.qh"
/*======================================================
TFORTMAP.QC Custom TeamFortress v3.1
(c) TeamFortress Software Pty Ltd 29/2/97
(c) William Kerney 2/23/00
(c) Craig Hauser 2/23/00
========================================================
Functions handling TeamFortress Map Entities
======================================================*/
// Prototypes
// Team Functions
void(float tno, float scoretoadd) TeamFortress_TeamIncreaseScore;
void(float all) TeamFortress_TeamShowScores;
float() TeamFortress_TeamGetWinner;
// Functions to handle entity placement when spawned
void() TF_PlaceItem;
void() TF_StartItem;
void() TF_PlaceGoal;
void() TF_StartGoal;
// Spawn functions for all Map Entities
float() CheckExistence;
void() info_tfdetect;
void() info_player_teamspawn;
void() info_tfgoal;
void() info_tfgoal_timer;
void() item_tfgoal;
// AutoDetection Function
void(entity AD) ParseTFDetect;
// Generic Functions
entity(float ino) Finditem;
entity(float gno) Findgoal;
entity(float gno) Findteamspawn;
void(entity Goal) InactivateGoal;
void(entity Goal) RestoreGoal;
void(entity Goal) RemoveGoal;
float(entity Goal, entity Player, entity AP) IsAffectedBy;
void(entity Goal, entity Player, entity AP, float addb) Apply_Results;
float(entity Goal, entity AP) APMeetsCriteria;
void(entity Goal) SetupRespawn;
void() DoRespawn;
void(entity Goal, entity AP) DoGoalWork;
void(entity Goal, entity AP) DoGroupWork;
void(entity Item, entity AP) DoItemGroupWork;
void(entity Goal, entity AP) DoTriggerWork;
void(entity Goal, entity AP, float addb) DoResults;
void(entity Goal, entity Player) RemoveResults;
//CH
void() tfgoalitem_dropthink2;
void() tfgoalitem_settouchandthink;
//CH 3 checks for a returning goalitem
void() tfgoalitem_checkoriginagain;
void() tfgoalitem_checkoriginagain2;
void() tfgoalitem_checkoriginagain3;
// Goal Functions
void() tfgoal_touch;
void() info_tfgoal_use;
// Timer Goal Functions
void() tfgoal_timer_tick;
// Item Functions
void() item_tfgoal_touch;
void(entity Item, entity AP, entity Goal) tfgoalitem_GiveToPlayer;
void(float Item)Update_team_with_flag_touch;
void(float Item)Update_team_with_flag_drop;
void(entity Item, entity AP, float method) tfgoalitem_RemoveFromPlayer;
void() tfgoalitem_remove;
void(entity Item) tfgoalitem_drop;
void(entity Item) tfgoalitem_checkgoalreturn;
void() ReturnItem;
void(entity Goal, entity Player, entity Item) DisplayItemStatus;
// CTF Support Functions
void() CTF_FlagCheck;
// Chris Support Functions
//void (float winner) RoundStop;
//=========================================================================
// ENTITY PLACEMENT HANDLING FUNCTIONS
//=========================================================================
// Copy the abbreviations into their storage areas
void(entity Goal) UpdateAbbreviations =
{
// The first time we enter this, copy the abbreviations
if (Goal.has_disconnected == FALSE)
{
// Floats
if (Goal.g_a != 0 && Goal.goal_activation == 0)
Goal.goal_activation = Goal.g_a;
if (Goal.g_e != 0 && Goal.goal_effects == 0)
Goal.goal_effects = Goal.g_e;
// Strings
if (Goal.t_s_h)
Goal.team_str_home = Goal.t_s_h;
if (Goal.t_s_m)
Goal.team_str_moved = Goal.t_s_m;
if (Goal.t_s_c)
Goal.team_str_carried = Goal.t_s_c;
if (Goal.n_s_h)
Goal.non_team_str_home = Goal.n_s_h;
if (Goal.n_s_m)
Goal.non_team_str_moved = Goal.n_s_m;
if (Goal.n_s_c)
Goal.non_team_str_carried = Goal.n_s_c;
if (Goal.b_b)
Goal.broadcast = Goal.b_b;
if (Goal.b_t)
Goal.team_broadcast = Goal.b_t;
if (Goal.b_n)
Goal.non_team_broadcast = Goal.b_n;
if (Goal.b_o)
Goal.owners_team_broadcast = Goal.b_o;
if (Goal.n_b)
Goal.netname_broadcast = Goal.n_b;
if (Goal.n_t)
Goal.netname_team_broadcast = Goal.n_t;
if (Goal.n_n)
Goal.netname_non_team_broadcast = Goal.n_n;
if (Goal.n_o)
Goal.netname_owners_team_broadcast = Goal.n_o;
if (Goal.d_t)
Goal.team_drop = Goal.d_t;
if (Goal.d_n)
Goal.non_team_drop = Goal.d_n;
if (Goal.d_n_t)
Goal.netname_team_drop = Goal.d_n_t;
if (Goal.d_n_n)
Goal.netname_non_team_drop = Goal.d_n_n;
Goal.has_disconnected = TRUE;
}
if (toggleflags & (TFLAG_WARSTANDARD | TFLAG_FLAGEMU))
{
local string newmdl;
if (toggleflags & TFLAG_WARSTANDARD)
newmdl = "progs/tf_stan.mdl";
else
newmdl = "progs/tf_flag.mdl";
switch (Goal.mdl)
{
case "progs/b_s_key.mdl":
case "progs/w_s_key.mdl":
case "progs/m_s_key.mdl":
Goal.skin = 1;
Goal.mdl = newmdl;
break;
case "progs/b_g_key.mdl":
case "progs/w_g_key.mdl":
case "progs/m_g_key.mdl":
Goal.skin = 2;
Goal.mdl = newmdl;
break;
}
}
};
// Place the Goal Item
void() TF_PlaceItem =
{
local vector temp1;
local vector temp2;
if (!toggleflags)
SetToggleFlags ();
UpdateAbbreviations (self);
// Graphic
if (self.mdl)
setmodel(self, self.mdl);
self.flags = FL_ITEM; // make extra wide
self.touch = item_tfgoal_touch;
setorigin(self, self.origin);
self.goal_state = TFGS_INACTIVE;
//CH my updates
if (self.goal_activation & TFGI_GOAL_TO_GROUND)
{
self.oldorigin = self.origin;
self.movetype = MOVETYPE_TOSS;
self.velocity_z = -40;
self.velocity_x = 0; //-50 + (random() * 100);
self.velocity_y = 0; //-50 + (random() * 100);
self.nextthink = time + 5.0; // give it 5 seconds
self.think = tfgoalitem_dropthink2; // and then find where it ended up
}
else
{
self.velocity = '0 0 0';
self.movetype = MOVETYPE_NONE;
self.oldorigin = self.origin;
}
//CH sets goal bounding box size
if (self.goal_min)
temp1 = self.goal_min;
else
temp1 = '-16 -16 -24';
if (self.goal_max)
temp2 = self.goal_max;
else
temp2 = '16 16 32';
setsize (self, temp1, temp2); //CH sets box size from above
if (self.goal_activation & TFGI_GOAL_IS_SOLID)
self.solid = SOLID_BBOX;
else
self.solid = SOLID_TRIGGER;
//CH end my updates
//On tfgoalitem_dropthink2
// self.velocity = '0 0 0';
// self.oldorigin = self.origin; // So we can return it later
if (self.goal_activation & TFGI_ITEMGLOWS)
self.effects = self.effects | EF_GlowColor(self);
// Setup the item_list mask
if (item_list_bit == 0)
item_list_bit = 1;
self.item_list = item_list_bit;
item_list_bit = item_list_bit * 2;
};
//=========================================================================
// Start the Goal Item
void() TF_StartItem =
{
self.nextthink = time + 0.2; // items start after other solids
self.think = TF_PlaceItem;
if (self.goal_state == TFGS_REMOVED)
RemoveGoal(self);
};
//=========================================================================
// Place the Goal
void() TF_PlaceGoal =
{
if (!toggleflags)
SetToggleFlags ();
UpdateAbbreviations (self);
if (self.classname != "info_tfgoal_timer")
{
// Only give touch functions to goals that can be activated by touch
if (self.goal_activation & TFGA_TOUCH)
self.touch = tfgoal_touch;
}
else
{
// Set up the next Timer Tick
self.think = tfgoal_timer_tick;
self.nextthink = time + self.search_time;
// So searches for this goal work later on
self.classname = "info_tfgoal";
}
self.flags = FL_ITEM; // make extra wide
self.movetype = MOVETYPE_NONE;
self.velocity = '0 0 0';
self.oldorigin = self.origin; // So we can return it later
};
//=========================================================================
// Start the Goal
void() TF_StartGoal =
{
self.nextthink = time + 0.2; // goals start after other solids
self.think = TF_PlaceGoal;
self.use = info_tfgoal_use;
if (self.goal_state == TFGS_REMOVED)
RemoveGoal(self);
};
//=========================================================================
// SPAWN FUNCTIONS
//=========================================================================
// Checks whether this entity should exist under the current settings
float() CheckExistence =
{
UpdateAbbreviations(self);
if (self.ex_skill_min && (skill <= self.ex_skill_min))
return FALSE;
else if (self.ex_skill_max && (skill >= self.ex_skill_max))
return FALSE;
return TRUE;
};
//=========================================================================
// Spawn a detection entity
void() info_tfdetect =
{
UpdateAbbreviations(self);
// The rest of the checking is done in the ParseTFDetect() function,
// which is called the first time someone enters the map, in client.qc
};
//=========================================================================
// Spawn a Team Spawn Point
void() info_player_teamspawn =
{
local entity te;
if (CheckExistence() == FALSE)
{
dremove(self);
return;
}
// find the highest team number
if (number_of_teams < self.team_no)
number_of_teams = self.team_no;
// Team spawnpoints must have a team associated with them
if (self.team_no <= 0)
{
RPrint("no team_no associated with info_player_teamspawn\n");
dremove(self);
}
// Does this give out a GoalItem?
if (self.items != 0)
{
te = Finditem(self.items);
if (!te)
{
RPrint("info_player_teamspawn specifies a GoalItem that does not exist\n");
dremove(self);
}
}
if (self.team_no == 1)
self.team_str_home = "ts1";
else if (self.team_no == 2)
self.team_str_home = "ts2";
else if (self.team_no == 3)
self.team_str_home = "ts3";
else if (self.team_no == 4)
self.team_str_home = "ts4";
};
void() i_p_t =
{
self.classname = "info_player_teamspawn";
info_player_teamspawn();
};
//=========================================================================
// Spawn a goal entity
void() info_tfgoal =
{
local vector temp1;
local vector temp2;
if (CheckExistence() == FALSE)
{
dremove(self);
return;
}
// Graphic
if (self.mdl)
{
// We try both, so at least one will work
precache_model(self.mdl);
precache_model2(self.mdl);
setmodel (self, self.mdl);
}
// Activation sound
if (self.noise)
{
// We try both, so at least one will work
precache_sound(self.noise);
precache_sound2(self.noise);
}
// For the powerups
precache_sound ("items/protect.wav");
precache_sound ("items/protect2.wav");
precache_sound ("items/protect3.wav");
precache_sound2 ("items/protect2.wav");
precache_sound2 ("items/protect3.wav");
precache_sound ("items/suit.wav");
precache_sound ("items/suit2.wav");
precache_sound ("items/inv1.wav");
precache_sound ("items/inv2.wav");
precache_sound ("items/inv3.wav");
precache_sound ("items/damage.wav");
precache_sound ("items/damage2.wav");
precache_sound ("items/damage3.wav");
//CH set solid state
if (self.goal_activation & TFGI_GOAL_IS_SOLID)
self.solid = SOLID_BBOX;
else
self.solid = SOLID_TRIGGER;
if (self.goal_state == 0)
self.goal_state = TFGS_INACTIVE;
//CH sets goal bounding box size
if (self.goal_min)
temp1 = self.goal_min;
else
temp1 = '-16 -16 -24';
if (self.goal_max)
temp2 = self.goal_max;
else
temp2 = '16 16 32';
setsize (self, temp1, temp2); //CH sets box size from above
TF_StartGoal ();
};
void() i_t_g =
{
self.classname = "info_tfgoal";
info_tfgoal();
};
//=========================================================================
// Spawn a Timer goal entity
void() info_tfgoal_timer =
{
if (CheckExistence() == FALSE)
{
dremove(self);
return;
}
// Graphic
if (self.mdl)
{
// We try both, so at least one will work
precache_model(self.mdl);
precache_model2(self.mdl);
setmodel(self, self.mdl);
}
// Activation sound
if (self.noise)
{
// We try both, so at least one will work
precache_sound(self.noise);
precache_sound2(self.noise);
}
// Timer Goals must have a time specified
if (self.search_time <= 0)
{
RPrint("Timer Goal created with no specified time.\n");
dremove(self);
}
self.solid = SOLID_NOT;
if (self.goal_state == 0)
self.goal_state = TFGS_INACTIVE;
setsize (self, '-16 -16 -24', '16 16 32');
TF_StartGoal ();
};
void() i_t_t =
{
self.classname = "info_tfgoal_timer";
info_tfgoal_timer();
};
//=========================================================================
// Spawn a goalitem entity
void() item_tfgoal =
{
local vector temp1;
local vector temp2;
if (CheckExistence() == FALSE)
{
dremove(self);
return;
}
UpdateAbbreviations (self);
// Graphic
if (self.mdl)
{
// We try both, so at least one will work
precache_model(self.mdl);
precache_model2(self.mdl);
setmodel(self, self.mdl);
}
else
{
// Default mdl for a GoalItem
self.mdl = "";
setmodel(self, "");
}
// Respawn sound
precache_sound2("items/itembk2.wav");
// Activation sound
if (self.noise)
{
// We try both, so at least one will work
precache_sound(self.noise);
precache_sound2(self.noise);
}
self.touch = item_tfgoal_touch;
if (self.goal_state == 0)
self.goal_state = TFGS_INACTIVE;
//CH sets how it is
if (self.goal_activation & TFGI_GOAL_IS_SOLID)
self.solid = SOLID_BBOX;
else
self.solid = SOLID_TRIGGER;
setorigin(self, self.origin);
if (!(self.netname))
self.netname = "goalitem";
if (self.pausetime <= 0)
self.pausetime = 60;
// for backwards compatability
if (self.delay != 0 && self.pausetime == 0)
self.pausetime = self.delay;
//CH sets goal bounding box size
if (self.goal_min)
temp1 = self.goal_min;
else
temp1 = '-16 -16 -24';
if (self.goal_max)
temp2 = self.goal_max;
else
temp2 = '16 16 32';
setsize (self, temp1, temp2); //CH sets box size from above
TF_StartItem ();
};
//=========================================================================
// AUTODETECTION FUNCTIONS
//=========================================================================
// Parse the Detection entities variables and set anything relevant
void(entity AD) ParseTFDetect =
{
// Check Version
/*
if (AD.broadcast)
{
if (AD.broadcast != "TeamFortress v2.5")
{
RPrint("This map was designed to be run on ");
RPrint(AD.broadcast);
RPrint("\nYou are using TeamFortress v2.5\n");
RPrint("You can get the latest version of this patch at: ");
RPrint("http://www.planetquake.com/teamfortress/\n");
}
}
*/
// Set the team menu string
if (AD.team_broadcast)
team_menu_string = AD.team_broadcast;
// Set toggleflags (DEFUNCT)
// toggleflags = AD.impulse;
// Do localcmds
localcmd(AD.message);
// Set life limits
team1lives = AD.ammo_shells;
team2lives = AD.ammo_nails;
team3lives = AD.ammo_rockets;
team4lives = AD.ammo_cells;
// If the lives of any of the teams are 0, they want infinite lives,
// so we set their number to -1
if (team1lives == 0)
team1lives = -1;
if (team2lives == 0)
team2lives = -1;
if (team3lives == 0)
team3lives = -1;
if (team4lives == 0)
team4lives = -1;
// Prevent hook use
if ((AD.hook_out == 1) || no_grapple == 1)
allow_hook = FALSE;
// Set player number limits for each team
team1maxplayers = AD.ammo_medikit;
team2maxplayers = AD.ammo_detpack;
team3maxplayers = AD.maxammo_medikit;
team4maxplayers = AD.maxammo_detpack;
if (team1maxplayers == 0)
team1maxplayers = 100;
if (team2maxplayers == 0)
team2maxplayers = 100;
if (team3maxplayers == 0)
team3maxplayers = 100;
if (team4maxplayers == 0)
team4maxplayers = 100;
// Set illegal playerclasses
illegalclasses = AD.playerclass;
illegalclasses1 = AD.maxammo_shells;
illegalclasses2 = AD.maxammo_nails;
illegalclasses3 = AD.maxammo_rockets;
illegalclasses4 = AD.maxammo_cells;
civilianteams = 0;
// Civilian team checking
if (illegalclasses1 == -1)
{
illegalclasses1 = 0;
civilianteams = civilianteams | TEAM1_CIVILIANS;
}
if (illegalclasses2 == -1)
{
illegalclasses2 = 0;
civilianteams = civilianteams | TEAM2_CIVILIANS;
}
if (illegalclasses3 == -1)
{
illegalclasses3 = 0;
civilianteams = civilianteams | TEAM3_CIVILIANS;
}
if (illegalclasses4 == -1)
{
illegalclasses4 = 0;
civilianteams = civilianteams | TEAM4_CIVILIANS;
}
};
//=========================================================================
// GENERIC FUNCTIONS
//=========================================================================
// Return the item with a goal_no equal to ino
entity(float ino) Finditem =
{
local entity tg;
local string st;
// Look for the goal
tg = find (NIL, classname, "item_tfgoal");
while (tg)
{
if (tg.goal_no == ino)
return tg;
tg = find(tg, classname, "item_tfgoal");
}
// Goal does not exist
RPrint("Could not find an item with a goal_no of ");
st = ftos(ino);
RPrint(st);
RPrint(".\n");
};
//=========================================================================
// Return the goal with a goal_no equal to gno
entity(float gno) Findgoal =
{
local entity tg;
local string st;
// Look for the goal
tg = find (NIL, classname, "info_tfgoal");
while (tg)
{
if (tg.goal_no == gno)
{
return tg;
}
tg = find(tg, classname, "info_tfgoal");
}
// Goal does not exist
RPrint("Could not find a goal with a goal_no of ");
st = ftos(gno);
RPrint(st);
RPrint(".\n");
};
//=========================================================================
// Return the TeamSpawn with a goal_no equal to gno
entity(float gno) Findteamspawn =
{
local entity tg;
local string st;
// Look for the goal
tg = find (NIL, classname, "info_player_teamspawn");
while (tg)
{
if (tg.goal_no == gno)
return tg;
tg = find(tg, classname, "info_player_teamspawn");
}
// Goal does not exist
RPrint("Could not find a Teamspawn with a goal_no of ");
st = ftos(gno);
RPrint(st);
RPrint(".\n");
};
//=========================================================================
// Inactivate a Timer/Goal
void(entity Goal) InactivateGoal =
{
#ifdef MAP_DEBUG
RPrint("Attempting to Inactivate ");
RPrint(Goal.netname);
#endif
if (Goal.goal_state == TFGS_ACTIVE)
{
// Not a timer goal
if (Goal.search_time == 0) {
if (Goal.goal_activation & TFGI_GOAL_IS_SOLID)
Goal.solid = SOLID_BBOX;
else
Goal.solid = SOLID_TRIGGER;
}
Goal.goal_state = TFGS_INACTIVE;
if (Goal.mdl)
setmodel(Goal, Goal.mdl);
}
#ifdef MAP_DEBUG
else
{
RPrint("... failed. Goal is ");
if (Goal.goal_state == TFGS_INACTIVE)
RPrint("inactive\n");
else if (Goal.goal_state == TFGS_REMOVED)
RPrint("removed\n");
else if (Goal.goal_state == TFGS_DELAYED)
RPrint("delayed\n");
}
#endif
};
//=========================================================================
void(entity Goal) RestoreGoal =
{
#ifdef MAP_DEBUG
RPrint("Attempting to Restore ");
RPrint(Goal.netname);
#endif
if (Goal.search_time == 0)
{
if (Goal.goal_activation & TFGI_GOAL_IS_SOLID)
Goal.solid = SOLID_BBOX;
else
Goal.solid = SOLID_TRIGGER;
if (Goal.mdl)
setmodel(Goal, Goal.mdl);
if (Goal.goal_state == TFGS_REMOVED)
Goal.goal_state = TFGS_INACTIVE;
}
if (Goal.goal_state == TFGS_REMOVED)
{
#ifdef MAP_DEBUG
RPrint("... succeeded.\n");
#endif
// Not a timer goal
if (Goal.search_time != 0)
// {
// if (Goal.goal_activation & TFGI_GOAL_IS_SOLID)
// Goal.solid = SOLID_BBOX;
// else
// Goal.solid = SOLID_TRIGGER;
// }
// else
Goal.nextthink = time + Goal.search_time;
Goal.goal_state = TFGS_INACTIVE;
// if (Goal.mdl)
// setmodel(Goal, Goal.mdl);
}
#ifdef MAP_DEBUG
else
{
RPrint("... failed. Goal is ");
if (Goal.goal_state == TFGS_INACTIVE)
RPrint("inactive\n");
else if (Goal.goal_state == TFGS_ACTIVE)
RPrint("active\n");
else if (Goal.goal_state == TFGS_DELAYED)
RPrint("delayed\n");
}
#endif
};
//=========================================================================
// Remove a Timer/Goal
void(entity Goal) RemoveGoal =
{
#ifdef MAP_DEBUG
RPrint("Removing ");
RPrint(Goal.netname);
RPrint("\n");
#endif
Goal.solid = SOLID_NOT;
Goal.goal_state = TFGS_REMOVED;
if (Goal.mdl)
setmodel(Goal, "");
};
//=========================================================================
// Return TRUE if the player is affected by the goal
float(entity Goal, entity Player, entity AP) IsAffectedBy =
{
local float genv;
// Don't affect anyone who isn't alive or is in Observer mode
if (Player.playerclass == PC_UNDEFINED)
return FALSE;
// if (Player.health <= 0)
// return FALSE;
// Same Environment Check
if (Goal.goal_effects & TFGE_SAME_ENVIRONMENT)
{
genv = pointcontents(Goal.origin);
if (pointcontents(Player.origin) != genv)
return FALSE;
}
if (Goal.t_length != 0)
{
// Within radius?
if (vlen(Goal.origin - Player.origin) <= Goal.t_length)
{
// Obstructed by walls?
if (Goal.goal_effects & TFGE_WALL)
{
traceline (Goal.origin, Player.origin, TRUE, Goal);
if (trace_fraction == 1)
return TRUE;
}
else
return TRUE;
}
}
if (Goal.classname != "info_tfgoal_timer")
{
if ((Goal.goal_effects & TFGE_AP) && (Player == AP))
return TRUE;
if ((Goal.goal_effects & TFGE_AP_TEAM) && (AP.team_no == Player.team_no))
return TRUE;
if ((Goal.goal_effects & TFGE_NOT_AP_TEAM) && (AP.team_no != Player.team_no))
return TRUE;
if ((Goal.goal_effects & TFGE_NOT_AP) && (Player != AP))
return TRUE;
}
if ((Goal.maxammo_shells != 0) && (Player.team_no == Goal.maxammo_shells))
return TRUE;
if ((Goal.maxammo_nails != 0) && (Player.team_no != Goal.maxammo_shells))
return TRUE;
return FALSE;
};
//=========================================================================
// Apply modifications to the Player passed in
void(entity Goal, entity Player, entity AP, float addb) Apply_Results =
{
local entity oldself, te, oldte;
stuffcmd(Player, "bf\n");
#ifdef MAP_DEBUG
RPrint("Applying Results from ");
RPrint(Goal.netname);
RPrint(" to ");
RPrint(AP.netname);
RPrint("\n");
#endif
// If this is a goalitem, record the fact that this player
// has been affected by it.
if (Goal.classname == "item_tfgoal")
Player.item_list = Player.item_list | Goal.item_list;
if (Player == AP)
{
// Increase the team score
if (Goal.count > 0)
{
if (Player.team_no > 0)
{
TeamFortress_TeamIncreaseScore(Player.team_no, Goal.count);
// Display short team scores
TeamFortress_TeamShowScores(2);
}
}
}
// Apply Stats, only if told to
if (addb)
{
#ifdef MAP_DEBUG
RPrint("Adding bonuses.\n");
#endif
// Some results are not applied to dead players
if (Player.health > 0)
{
if (Goal.health > 0)
T_Heal(Player, Goal.health, 0);
if (Goal.health < 0)
{
if (Goal.invincible_finished < 0)
{
// KK allow goal to strip invinc cheats
Player.items = Player.items & ~IT_INVULNERABILITY;
Player.invincible_time = 0;
Player.invincible_finished = 0;
}
// Make sure we don't gib them. We don't want to gib, because
// it creates too many entities if a lot of players are affected
// by this Goal.
if ((0 - Player.health) > Goal.health)
TF_T_Damage (Player, Goal, Goal, (Player.health + 1), TF_TD_IGNOREARMOUR, TF_TD_OTHER);
else
TF_T_Damage (Player, Goal, Goal, (0 - Goal.health), TF_TD_IGNOREARMOUR, TF_TD_OTHER);
}
}
// The player may be dead now, so check again
if (Player.health > 0)
{
if (Goal.armortype)
Player.armortype = Goal.armortype;
Player.armorvalue = Player.armorvalue + Goal.armorvalue;
if (Goal.armorclass)
{ //WK Modify to | it with bought armor
//TODO: Make this a map specific option
//WK Don't give wooden armor since it doesn't exist
// any more. And we don't want gel for free, right?
Player.armorclass = Goal.armorclass - (Goal.armorclass & AT_SAVEMELEE);
if (Player.tf_items & NIT_KEVLAR)
Player.armorclass = Player.armorclass | AT_SAVESHOT;
if (Player.tf_items & NIT_GEL)
Player.armorclass = Player.armorclass | AT_SAVEMELEE;
if (Player.tf_items & NIT_BLAST)
Player.armorclass = Player.armorclass | AT_SAVEEXPLOSION;
if (Player.tf_items & NIT_CERAMIC)
Player.armorclass = Player.armorclass | AT_SAVEELECTRICITY;
if (Player.tf_items & NIT_ASBESTOS)
Player.armorclass = Player.armorclass | AT_SAVEFIRE;
}
Player.ammo_shells = Player.ammo_shells + Goal.ammo_shells;
Player.ammo_nails = Player.ammo_nails + Goal.ammo_nails;
Player.ammo_rockets = Player.ammo_rockets + Goal.ammo_rockets;
Player.ammo_cells = Player.ammo_cells + Goal.ammo_cells;
Player.ammo_medikit = Player.ammo_medikit + Goal.ammo_medikit;
Player.ammo_detpack = Player.ammo_detpack + Goal.ammo_detpack;
if (Player.tp_grenades_1 != 0) //WK Stop Bug grenades
Player.no_grenades_1 = Player.no_grenades_1 + Goal.no_grenades_1;
if (Player.tp_grenades_2 != 0) //WK Stop Bug grenades
Player.no_grenades_2 = Player.no_grenades_2 + Goal.no_grenades_2;
if (Player.no_grenades_1 > GetMaxGrens(Player,1))
Player.no_grenades_1 = GetMaxGrens(Player,1);
if (Player.no_grenades_2 > GetMaxGrens(Player,2))
Player.no_grenades_2 = GetMaxGrens(Player,2);
/*if (Player.no_grenades_1 > 4)
if (Player.tf_items & NIT_AMMO_BANDOLIER)
Player.no_grenades_1 = 5;
else
Player.no_grenades_1 = 4;
if (Player.no_grenades_2 > 4)
if (Player.tf_items & NIT_AMMO_BANDOLIER)
Player.no_grenades_2 = 5;
else
Player.no_grenades_2 = 4;*/
if (Player.ammo_detpack > Player.maxammo_detpack)
Player.ammo_detpack = Player.maxammo_detpack;
// Apply any powerups
if (Goal.invincible_finished > 0)
{
Player.items = Player.items | IT_INVULNERABILITY;
Player.invincible_time = 1;
Player.invincible_finished = time + Goal.invincible_finished;
// if its a GoalItem, powerup is permanent, so we use TFSTATE flags
if (Goal.classname == "item_tfgoal")
{
Player.tfstate = Player.tfstate | TFSTATE_INVINCIBLE;
Player.invincible_finished = time + 666;
}
}
if (Goal.invisible_finished > 0)
{
Player.items = Player.items | IT_INVISIBILITY;
Player.invisible_time = 1;
Player.invisible_finished = time + Goal.invisible_finished;
// if its a GoalItem, powerup is permanent, so we use TFSTATE flags
if (Goal.classname == "item_tfgoal")
{
Player.tfstate = Player.tfstate | TFSTATE_INVISIBLE;
Player.invisible_finished = time + 666;
}
}
if (Goal.super_damage_finished > 0)
{
//WK Remove Inspiration when we get the real thing
Player.tfstate = Player.tfstate - (Player.tfstate & TFSTATE_INSPIRED);
Player.items = Player.items | IT_QUAD;
Player.super_time = 1;
Player.super_damage_finished = time + Goal.super_damage_finished;
// if its a GoalItem, powerup is permanent, so we use TFSTATE flags
if (Goal.classname == "item_tfgoal")
{
Player.tfstate = Player.tfstate | TFSTATE_QUAD;
Player.super_damage_finished = time + 666;
}
}
if (Goal.radsuit_finished > 0)
{
Player.items = Player.items | IT_SUIT;
Player.rad_time = 1;
Player.radsuit_finished = time + Goal.radsuit_finished;
// if its a GoalItem, powerup is permanent, so we use TFSTATE flags
if (Goal.classname == "item_tfgoal")
{
Player.tfstate = Player.tfstate | TFSTATE_RADSUIT;
Player.radsuit_finished = time + 666;
}
}
}
// These results are applied to dead and living players
if (Goal.lives < 0)
bprint(PRINT_HIGH,"ERROR: GOAL GAVE OUT NEGATIVE LIVES. CUSTOMTF DOESN'T SUPPORT THAT YET\n");
Player.lives = Player.lives + Goal.lives;
RPrint("Giving " + ftos(Goal.frags) + " frags to " + Player.netname +" from goal " + Goal.netname + ".\n");
Player.real_frags = Player.real_frags + Goal.frags;
if (!(toggleflags & TFLAG_TEAMFRAGS))
Player.frags = Player.real_frags;
// Now apply the Playerclass limitations & Redisplay Ammo counts
oldself = self;
self = Player;
TeamFortress_CheckClassStats();
//WK Fix CH's bug fix with this if statement :)
if (!(self.current_weapon == WEAP_ASSAULT_CANNON && self.heat >= 0))
W_SetCurrentAmmo(); //CH stops interrupt weapon fire bug
self = oldself;
}
#ifdef MAP_DEBUG
else
{
RPrint("NOT Adding bonuses.\n");
}
#endif
if (Goal.goal_result & TFGR_CAUSERESPAWN && Player)
{
if ((Player.playerclass == PC_CUSTOM && Player.done_custom & CUSTOM_FINISHED) || (Player.playerclass != PC_UNDEFINED && Player.playerclass != PC_CUSTOM)) //CH respawn player
{
//CH this needs lots of testing
//*CH
local entity temp;
temp = self;
self = Player;
PutClientInServer();
self = temp;
}
}
// If the Goal resets Spy skin/color then do it
//CH added looking for spykit also removes thief
if (Goal.goal_result & TFGR_REMOVE_DISGUISE)
{
if (Player.job & JOB_THIEF && (Player.job & JOB_ACTIVE || Player.job & JOB_FULL_HIDE))
RevealThief(Player,FALSE);
if (Player.playerclass == PC_SPY || Player.cutf_items & CUTF_SPY_KIT)
{
makeImmune(self,time + 4);
//self.immune_to_check = time + 4;
Spy_RemoveDisguise(Player);
}
}
// If there's a GoalItem for this goal, give it to the player
// GoalItems use "items" for the console lights... so don't do it for items.
if (Goal.items != 0 && Goal.classname != "item_tfgoal")
{
// Find the item
te = Finditem(Goal.items);
if (te)
tfgoalitem_GiveToPlayer(te, Player, Goal);
}
// If this goal removes an item from the player, remove it
if (Goal.axhitme != 0)
{
te = Finditem(Goal.axhitme);
if (te.owner == Player)
{
tfgoalitem_RemoveFromPlayer(te, Player, 1);
}
}
// if this goal removes a group of items from the player, remove them
if (Goal.remove_item_group != 0)
{
// Find all goals
te = find (NIL, classname, "item_tfgoal");
while (te)
{
if (te.group_no == Goal.remove_item_group && te.owner == AP)
{
// messy, since we need the item to find the next one,
// and we also want to remove it
oldte = te;
te = find(te, classname, "item_tfgoal");
tfgoalitem_RemoveFromPlayer(oldte, Player, 1);
}
else
{
te = find(te, classname, "item_tfgoal");
}
}
}
// If this goal displays some Item statuses, then do so
if (Goal.display_item_status1 != 0)
{
te = Finditem(Goal.display_item_status1);
if (te)
DisplayItemStatus(Goal, Player, te);
else
sprint(Player, PRINT_HIGH, "Item is missing.\n");
}
if (Goal.display_item_status2 != 0)
{
te = Finditem(Goal.display_item_status2);
if (te)
DisplayItemStatus(Goal, Player, te);
else
sprint(Player, PRINT_HIGH, "Item is missing.\n");
}
if (Goal.display_item_status3 != 0)
{
te = Finditem(Goal.display_item_status3);
if (te)
DisplayItemStatus(Goal, Player, te);
else
sprint(Player, PRINT_HIGH, "Item is missing.\n");
}
if (Goal.display_item_status4 != 0)
{
te = Finditem(Goal.display_item_status4);
if (te)
DisplayItemStatus(Goal, Player, te);
else
sprint(Player, PRINT_HIGH, "Item is missing.\n");
}
};
//=========================================================================
// Remove any results applied to this player by the Goal
// Used when a GoalItem is dropped/removed
void (entity Goal, entity Player) RemoveResults =
{
local entity oldself, te;
local float puinvin, puinvis, puquad, purad;
// Only remove the stats if the player has been affected
// by this item. This is needed because the player may have
// died since being affected
if (Goal.classname == "item_tfgoal")
{
if (!(Player.item_list & Goal.item_list))
return;
if (Goal.goal_activation & TFGI_DONTREMOVERES)
return;
// Remove the affected flag
Player.item_list = Player.item_list - (Player.item_list & Goal.item_list);
}
if (Goal.health > 0)
TF_T_Damage (Player, Goal, Goal, Goal.health, TF_TD_IGNOREARMOUR, TF_TD_OTHER);
if (Goal.health < 0)
T_Heal(Player, (0 - Goal.health), 0);
Player.lives = Player.lives - Goal.lives;
Player.armortype = Player.armortype - Goal.armortype;
Player.armorvalue = Player.armorvalue - Goal.armorvalue;
Player.armorclass = Player.armorclass - (Player.armorclass & Goal.armorclass);
Player.real_frags = Player.real_frags - Goal.frags;
if (!(toggleflags & TFLAG_TEAMFRAGS))
Player.frags = Player.real_frags;
Player.ammo_shells = Player.ammo_shells - Goal.ammo_shells;
Player.ammo_nails = Player.ammo_nails - Goal.ammo_nails;
Player.ammo_rockets = Player.ammo_rockets - Goal.ammo_rockets;
Player.ammo_cells = Player.ammo_cells - Goal.ammo_cells;
Player.ammo_medikit = Player.ammo_medikit - Goal.ammo_medikit;
Player.ammo_detpack = Player.ammo_detpack - Goal.ammo_detpack;
Player.no_grenades_1 = Player.no_grenades_1 - Goal.no_grenades_1;
Player.no_grenades_2 = Player.no_grenades_2 - Goal.no_grenades_2;
puinvin = FALSE;
puinvis = FALSE;
puquad = FALSE;
purad = FALSE;
// Make sure we don't remove an effect another Goal is also supplying
te = find (NIL, classname, "item_tfgoal");
while (te)
{
if ((te.owner == Player) && (te != Goal))
{
if (te.invincible_finished > 0)
puinvin = TRUE;
if (te.invisible_finished > 0)
puinvis = TRUE;
if (te.super_damage_finished > 0)
puquad = TRUE;
if (te.radsuit_finished > 0)
purad = TRUE;
}
te = find(te, classname, "item_tfgoal");
}
// Remove all powerups
if ((Goal.invincible_finished > 0) && (!puinvin))
{
// if its a GoalItem, powerup was permanent, so we remove TFSTATE flag
Player.tfstate = Player.tfstate - (Player.tfstate & TFSTATE_INVINCIBLE);
Player.items = Player.items | IT_INVULNERABILITY;
Player.invincible_time = 1;
Player.invincible_finished = time + Goal.invincible_finished;
}
if ((Goal.invisible_finished > 0) && (!puinvis))
{
// if its a GoalItem, powerup was permanent, so we remove TFSTATE flag
Player.tfstate = Player.tfstate - (Player.tfstate & TFSTATE_INVISIBLE);
Player.items = Player.items | IT_INVISIBILITY;
Player.invisible_time = 1;
Player.invisible_finished = time + Goal.invisible_finished;
}
if ((Goal.super_damage_finished > 0) && (!puquad))
{
// if its a GoalItem, powerup was permanent, so we remove TFSTATE flag
Player.tfstate = Player.tfstate - (Player.tfstate & TFSTATE_QUAD);
Player.items = Player.items | IT_QUAD;
Player.super_time = 1;
Player.super_damage_finished = time + Goal.super_damage_finished;
}
//WK Don't remove the scuba gear
if ((Goal.radsuit_finished > 0) && (!purad) && !(Player.tf_items & NIT_SCUBA))
{
// if its a GoalItem, powerup was permanent, so we remove TFSTATE flag
Player.tfstate = Player.tfstate - (Player.tfstate & TFSTATE_RADSUIT);
Player.items = Player.items | IT_SUIT;
Player.rad_time = 1;
Player.radsuit_finished = time + Goal.radsuit_finished;
}
// Now apply the Playerclass limitations & Redisplay Ammo counts
oldself = self;
self = Player;
TeamFortress_CheckClassStats();
W_SetCurrentAmmo ();
self = oldself;
};
//=========================================================================
// Return TRUE if the player meets the AP criteria
float(entity Goal, entity AP) APMeetsCriteria =
{
local float gotone, temp;
local entity te;
#ifdef MAP_DEBUG
RPrint("==========================\n");
#ifdef VERBOSE
RPrint("AP Criteria Checking\n");
RPrint("Goal: ");
#else
RPrint("APCriteria: ");
#endif
RPrint(Goal.netname);
#endif
if (AP)
{
#ifdef MAP_DEBUG
#ifdef VERBOSE
RPrint("\nAP : ");
#else
RPrint(" by ");
#endif
RPrint(AP.netname);
RPrint("\n");
RPrint(" Checking team.");
#endif
// If a player of a specific team can only activate this
if (Goal.team_no)
{
if (Goal.team_no != AP.team_no)
return FALSE;
}
#ifdef MAP_DEBUG
#ifdef VERBOSE
RPrint("passed.\n Checking class...");
#else
RPrint("class.");
#endif
#endif
//CH uses function in tfort.qc to return the class for evaluation
if (AP.playerclass == PC_CUSTOM && (Goal.playerclass))
{
temp = Return_Custom_Skins(AP);
//CH now that we have their 'class' check
if (Goal.playerclass != temp)
return FALSE;
}
// If a player of a specific class can only activate this
else if (Goal.playerclass)
{
if (Goal.playerclass != AP.playerclass)
return FALSE;
}
#ifdef MAP_DEBUG
#ifdef VERBOSE
RPrint("passed.\n Checking items...");
#else
RPrint("items.");
#endif
#endif
// If this activation needs a GoalItem, make sure the player has it
if (Goal.items_allowed)
{
te = Finditem(Goal.items_allowed);
if (te.owner != AP)
return FALSE;
}
#ifdef MAP_DEBUG
#ifdef VERBOSE
RPrint("passed.\n Checking goal states...");
#else
RPrint("goal states.");
#endif
#endif
}
#ifdef MAP_DEBUG
#ifdef VERBOSE
if (AP)
RPrint("passed.\n Checking Goal States...");
else
RPrint("\nChecking Goal States...");
#else
RPrint("Goal States.");
#endif
#endif
// Check Goal states
if (Goal.if_goal_is_active)
{
te = Findgoal(Goal.if_goal_is_active);
if (te.goal_state != TFGS_ACTIVE)
return FALSE;
}
if (Goal.if_goal_is_inactive)
{
te = Findgoal(Goal.if_goal_is_inactive);
if (te.goal_state != TFGS_INACTIVE)
return FALSE;
}
if (Goal.if_goal_is_removed)
{
te = Findgoal(Goal.if_goal_is_removed);
if (te.goal_state != TFGS_REMOVED)
return FALSE;
}
#ifdef MAP_DEBUG
#ifdef VERBOSE
RPrint("passed.\n Checking group states...");
#else
RPrint("group states.");
#endif
#endif
if (Goal.if_group_is_active)
{
// Find all goals in the group
te = find (NIL, classname, "info_tfgoal");
while (te)
{
if (te.group_no == Goal.if_group_is_active)
{
if (te.goal_state != TFGS_ACTIVE)
return FALSE;
}
te = find(te, classname, "info_tfgoal");
}
}
if (Goal.if_group_is_inactive)
{
// Find all goals in the group
te = find (NIL, classname, "info_tfgoal");
while (te)
{
if (te.group_no == Goal.if_group_is_inactive)
{
if (te.goal_state != TFGS_INACTIVE)
return FALSE;
}
te = find(te, classname, "info_tfgoal");
}
}
if (Goal.if_group_is_removed)
{
// Find all goals in the group
te = find (NIL, classname, "info_tfgoal");
while (te)
{
if (te.group_no == Goal.if_group_is_removed)
{
if (te.goal_state != TFGS_REMOVED)
return FALSE;
}
te = find(te, classname, "info_tfgoal");
}
}
#ifdef MAP_DEBUG
#ifdef VERBOSE
RPrint("passed.\n Checking item states...");
#else
RPrint("item states.");
#endif
#endif
if (Goal.if_item_has_moved)
{
// Find the item
te = Finditem(Goal.if_item_has_moved);
if (te)
{
if (te.goal_state != TFGS_ACTIVE && te.origin == te.oldorigin )
return FALSE;
}
}
if (Goal.if_item_hasnt_moved)
{
// Find the item
te = Finditem(Goal.if_item_hasnt_moved);
if (te)
{
if (te.goal_state == TFGS_ACTIVE || te.origin != te.oldorigin )
return FALSE;
}
}
#ifdef MAP_DEBUG
#ifdef VERBOSE
RPrint("passed.\n Checking item groups...");
#else
RPrint("item groups ");
#endif
#endif
if (AP)
{
gotone = FALSE;
if (Goal.has_item_from_group)
{
// Find all goals
te = find (NIL, classname, "item_tfgoal");
while (te && !gotone)
{
if (te.group_no == Goal.has_item_from_group && te.owner == AP)
gotone = TRUE;
te = find(te, classname, "item_tfgoal");
}
if (!gotone)
return FALSE;
}
}
#ifdef MAP_DEBUG
RPrint("passed.\n");
#endif
return TRUE;
};
//=========================================================================
// Setup the way this Timer/Goal/Item will respawn
void(entity Goal) SetupRespawn =
{
// Timer Goal?
if (Goal.search_time != 0)
{
InactivateGoal(Goal);
Goal.think = tfgoal_timer_tick;
Goal.nextthink = time + Goal.search_time;
return;
}
// Check status of respawn for this goal
// Single Activation, do nothing
if (Goal.goal_result & TFGR_SINGLE)
{
RemoveGoal(Goal);
return;
}
// Respawn Activation, set up respawn
if (Goal.wait > 0)
{
Goal.nextthink = time + Goal.wait;
Goal.think = DoRespawn;
if (Goal.mdl) //CH if it has mdl, remove it so
setmodel(Goal, ""); //that people dont think it works
return;
}
// Permanently active goal?
else if (Goal.wait == -1)
return;
// Otherwise, it's a Multiple Goal
InactivateGoal(Goal);
};
//=========================================================================
// Respawn the goal
void() DoRespawn =
{
RestoreGoal(self);
InactivateGoal(self);
};
//=========================================================================
// THE Function for knowing whether to activate or not
// Fuck this map code is a mess!
// Returns TRUE if you should activate, FALSE if not
float(entity Goal, entity AP) Activated =
{
local float APMet, RevAct, Act;
#ifdef MAP_DEBUG
#ifdef VERBOSE
RPrint("\nDoIActivate: ");
#else
RPrint("\nGoal: ");
#endif
if (!Goal.netname)
RPrint(Goal.classname);
else
RPrint(Goal.netname);
RPrint(", AP: ");
RPrint(AP.netname);
RPrint("\n");
#endif
if (Goal.goal_state == TFGS_ACTIVE)
{
#ifdef MAP_DEBUG
#ifdef VERBOSE
RPrint("-- Goal already active --\n");
#endif
#endif
return FALSE;
}
if (Goal.goal_state == TFGS_REMOVED)
{
#ifdef MAP_DEBUG
#ifdef VERBOSE
RPrint("-- Goal is in Removed state --\n");
#endif
#endif
return FALSE;
}
if (Goal.goal_state == TFGS_DELAYED)
{
#ifdef MAP_DEBUG
#ifdef VERBOSE
RPrint("-- Goal is being Delayed --\n");
#endif
#endif
return FALSE;
}
APMet = APMeetsCriteria(Goal, AP);
if (Goal.classname == "item_tfgoal")
RevAct = Goal.goal_activation & TFGI_REVERSE_AP;
else
RevAct = Goal.goal_activation & TFGA_REVERSE_AP;
Act = FALSE;
// Does the AP match the AP Criteria?
if (APMet)
{
#ifdef MAP_DEBUG
RPrint("\n");
#ifdef VERBOSE
RPrint("-- Criteria met --\n");
#endif
#endif
if (!RevAct)
Act = TRUE;
}
else
{
#ifdef MAP_DEBUG
RPrint("\n");
#ifdef VERBOSE
RPrint("-- Criteria not met --\n");
#endif
#endif
if (RevAct)
{
Act = TRUE;
#ifdef MAP_DEBUG
#ifdef VERBOSE
RPrint("Reverse \n");
#endif
#endif
}
}
#ifdef MAP_DEBUG
#ifdef VERBOSE
if (Act)
{
RPrint("Activation.\n");
}
else
{
RPrint("NO Activation.\n");
}
#endif
#endif
return Act;
};
//=========================================================================
// Attempt to activate a Goal
void(entity Goal, entity AP, entity ActivatingGoal) AttemptToActivate =
{
local entity te;
if (Activated(Goal, AP))
{
// It's all cool. Do the Results.
if (ActivatingGoal == Goal)
DoResults(Goal, AP, TRUE);
else if (ActivatingGoal)
DoResults(Goal, AP, (ActivatingGoal.goal_result & TFGR_ADD_BONUSES));
else
DoResults(Goal, AP, 0);
}
else
{
// If an else goal should be activated, activate it
if (Goal.else_goal != 0)
{
#ifdef MAP_DEBUG
#ifdef VERBOSE
RPrint(" Else Goal.\n");
#endif
#endif
te = Findgoal(Goal.else_goal);
if (te)
AttemptToActivate(te, AP, ActivatingGoal);
}
}
};
//=========================================================================
// Do all the activation/inactivation/etc of individual Goals
void(entity Goal, entity AP) DoGoalWork =
{
local entity te, RI;
// If another goal should be activated, activate it
if (Goal.activate_goal_no != 0)
{
Update_team_with_flag_touch(Goal.activate_goal_no);
te = Findgoal(Goal.activate_goal_no);
if (te)
AttemptToActivate(te, AP, Goal);
}
// If another goal should be inactivated, inactivate it
if (Goal.inactivate_goal_no != 0)
{
Update_team_with_flag_drop(Goal.inactivate_goal_no);
te = Findgoal(Goal.inactivate_goal_no);
if (te)
InactivateGoal(te);
}
// If another goal should be restored, restore it
if (Goal.restore_goal_no != 0)
{
te = Findgoal(Goal.restore_goal_no);
if (te)
RestoreGoal(te);
}
// If another goal should be removed, remove it
if (Goal.remove_goal_no != 0)
{
te = Findgoal(Goal.remove_goal_no);
if (te)
RemoveGoal(te);
}
// If a GoalItem should be returned, return it
if (Goal.return_item_no != 0)
{
te = Finditem(Goal.return_item_no);
if (te)
{
if (te.goal_state == TFGS_ACTIVE)
tfgoalitem_RemoveFromPlayer(te, te.owner, 1);
RI = spawn();
RI.enemy = te;
RI.think = ReturnItem;
RI.nextthink = time + 0.1;
te.solid = SOLID_NOT;
}
}
// Spawnpoint behaviour
if (Goal.remove_spawnpoint != 0)
{
te = Findteamspawn(Goal.remove_spawnpoint);
if (te)
{
te.goal_state = TFGS_REMOVED;
te.team_str_home = "";
}
}
if (Goal.restore_spawnpoint != 0)
{
te = Findteamspawn(Goal.restore_spawnpoint);
if (te)
{
if (te.goal_state == TFGS_REMOVED)
{
te.goal_state = TFGS_INACTIVE;
if (te.team_no == 1)
te.team_str_home = "ts1";
else if (te.team_no == 2)
te.team_str_home = "ts2";
else if (te.team_no == 3)
te.team_str_home = "ts3";
else if (te.team_no == 4)
te.team_str_home = "ts4";
}
}
}
};
//=========================================================================
// Do all the activation/inactivation/etc of Goal Groups
void(entity Goal, entity AP) DoGroupWork =
{
local string st;
local entity tg;
local float allset;
// Check all goals activated flag
if (Goal.all_active != 0)
{
if (Goal.last_impulse == 0)
{
// No goal specified in .lastimpulse. Print error.
RPrint("Goal ");
st = ftos(Goal.goal_no);
RPrint(st);
RPrint(" has a .all_active specified, but no .last_impulse\n");
}
else
{
#ifdef MAP_DEBUG
#ifdef VERBOSE
RPrint("All Active Group Check.\n");
#endif
#endif
allset = 1;
// Find all goals
tg = find (NIL, classname, "info_tfgoal");
while (tg)
{
if (tg.group_no == Goal.all_active)
{
if (tg.goal_state != TFGS_ACTIVE)
allset = 0;
}
tg = find(tg, classname, "info_tfgoal");
}
// If all goals in this group are activated, do it
if (allset)
{
#ifdef MAP_DEBUG
#ifdef VERBOSE
RPrint("All Active, Activating last_impulse.\n");
#endif
#endif
tg = Findgoal(Goal.last_impulse);
if (tg)
DoResults(tg, AP, (Goal.goal_result & TFGR_ADD_BONUSES));
}
#ifdef MAP_DEBUG
#ifdef VERBOSE
else
{
RPrint("Not all Active.\n");
}
#endif
#endif
}
}
// Check Activate all in the group flag
if (Goal.activate_group_no != 0)
{
// Find all goals
tg = find (NIL, classname, "info_tfgoal");
while (tg)
{
if (tg.group_no == Goal.activate_group_no)
DoResults(tg, AP, 0); // Don't apply bonuses
tg = find(tg, classname, "info_tfgoal");
}
}
// Check Inactivate all in the group flag
if (Goal.inactivate_group_no != 0)
{
// Find all goals
tg = find (NIL, classname, "info_tfgoal");
while (tg)
{
if (tg.group_no == Goal.inactivate_group_no)
InactivateGoal(tg);
tg = find(tg, classname, "info_tfgoal");
}
}
// Check Remove all in the group flag
if (Goal.remove_group_no != 0)
{
// Find all goals
tg = find (NIL, classname, "info_tfgoal");
while (tg)
{
if (tg.group_no == Goal.remove_group_no)
RemoveGoal(tg);
tg = find(tg, classname, "info_tfgoal");
}
}
// Check Restore all in the group flag
if (Goal.restore_group_no != 0)
{
// Find all goals
tg = find (NIL, classname, "info_tfgoal");
while (tg)
{
if (tg.group_no == Goal.restore_group_no)
RestoreGoal(tg);
tg = find(tg, classname, "info_tfgoal");
}
}
// Spawnpoint behaviour
if (Goal.remove_spawngroup != 0)
{
// Find all goals
tg = find (NIL, classname, "info_player_teamspawn");
while (tg)
{
if (tg.group_no == Goal.remove_spawngroup)
{
tg.goal_state = TFGS_REMOVED;
tg.team_str_home = "";
}
tg = find(tg, classname, "info_player_teamspawn");
}
}
if (Goal.restore_spawngroup != 0)
{
// Find all goals
tg = find (NIL, classname, "info_player_teamspawn");
while (tg)
{
if (tg.group_no == Goal.restore_spawngroup)
{
tg.goal_state = TFGS_INACTIVE;
if (tg.team_no == 1)
tg.team_str_home = "ts1";
else if (tg.team_no == 2)
tg.team_str_home = "ts2";
else if (tg.team_no == 3)
tg.team_str_home = "ts3";
else if (tg.team_no == 4)
tg.team_str_home = "ts4";
}
tg = find(tg, classname, "info_player_teamspawn");
}
}
};
//=========================================================================
// Do all the checking of Item Groups
void(entity Item, entity AP) DoItemGroupWork =
{
local entity tg, carrier;
local float allcarried;
local string st;
allcarried = TRUE;
if (Item.distance != 0)
{
if (Item.pain_finished == 0)
{
// No goal specified in .pain_finished. Print error.
RPrint("GoalItem ");
st = ftos(Item.goal_no);
RPrint(st);
RPrint(" has a .distance specified, but no .pain_finished\n");
}
// Find all goals
tg = find (NIL, classname, "item_tfgoal");
while (tg)
{
if (tg.group_no == Item.distance)
{
if (tg.goal_state != TFGS_ACTIVE)
allcarried = FALSE;
}
tg = find(tg, classname, "item_tfgoal");
}
if (allcarried == TRUE)
{
tg = Findgoal(Item.pain_finished);
if (tg)
DoResults(tg, AP, (Item.goal_result & TFGR_ADD_BONUSES));
}
}
allcarried = TRUE;
if (Item.speed != 0)
{
if (Item.attack_finished == 0)
{
// No goal specified in .attack_finished. Print error.
RPrint("GoalItem ");
st = ftos(Item.goal_no);
RPrint(st);
RPrint(" has a .speed specified, but no .attack_finished\n");
}
carrier = NIL;
// Find all goals
tg = find (NIL, classname, "item_tfgoal");
while (tg)
{
if (tg.group_no == Item.speed)
{
if (tg.goal_state != TFGS_ACTIVE)
allcarried = FALSE;
else if (!carrier)
carrier = tg.owner;
else if (carrier != tg.owner)
allcarried = FALSE;
}
tg = find(tg, classname, "item_tfgoal");
}
if (allcarried == TRUE)
{
tg = Findgoal(Item.attack_finished);
if (tg)
DoResults(tg, AP, (Item.goal_result & TFGR_ADD_BONUSES));
}
}
};
//=========================================================================
// Do all the activation/removal of Quake Triggers
void(entity Goal, entity AP) DoTriggerWork =
{
local entity otemp, stemp, t;
#ifdef MAP_DEBUG
if (Goal.killtarget)
{
RPrint("Killing Target(s): ");
RPrint(Goal.killtarget);
RPrint("\n");
}
#endif
// remove targets
if (Goal.killtarget)
{
t = NIL;
do
{
t = find(t, targetname, Goal.killtarget);
if (t && t != Goal)
remove(t);
} while ( t );
}
#ifdef MAP_DEBUG
if (Goal.target)
{
RPrint("Activating Target(s): ");
RPrint(Goal.target);
RPrint("\n");
}
#endif
// fire targets
if (Goal.target)
{
t = NIL;
activator = AP;
do
{
t = find (t, targetname, Goal.target);
if (!t)
return;
stemp = self;
otemp = other;
self = t;
other = stemp;
if (self.use)
self.use ();
self = stemp;
other = otemp;
activator = AP;
} while ( t );
}
};
//=========================================================================
// Handles Delayed Activation of Goals
void() DelayedResult =
{
if (self.enemy.goal_state == TFGS_DELAYED)
DoResults(self.enemy, self.owner, self.weapon);
dremove(self);
};
//=========================================================================
// Do the results for the Timer/Goal/Item
void(entity Goal, entity AP, float addb) DoResults =
{
local entity te;
local float winners;
// Is the goal already activated?
// This check is needed for goals which are being activated by other goals
if (Goal.goal_state == TFGS_ACTIVE)
return;
// Delayed Activation?
if (Goal.delay_time > 0 && Goal.goal_state != TFGS_DELAYED)
{
#ifdef MAP_DEBUG
RPrint("Delaying Results of ");
RPrint(Goal.netname);
RPrint("\n");
#endif
Goal.goal_state = TFGS_DELAYED;
te = spawn();
te.think = DelayedResult;
te.nextthink = time + Goal.delay_time;
te.owner = AP;
te.enemy = Goal;
te.weapon = addb;
return;
}
UpdateAbbreviations(Goal);
Goal.goal_state = TFGS_INACTIVE;
#ifdef MAP_DEBUG
#ifdef VERBOSE
RPrint("-= Doing Results =-\n");
RPrint("Goal: ");
#else
RPrint("DoRes: ");
#endif
RPrint(Goal.netname);
#ifdef VERBOSE
RPrint("\nAP : ");
#else
RPrint(" by ");
#endif
RPrint(AP.netname);
if (addb)
RPrint("\n adding bonuses");
else
RPrint("\nNOT adding bonuses");
#ifdef VERBOSE
RPrint("\n-=================-\n");
#else
RPrint("\n");
#endif
#endif
//- OfN if Goal.option is 1 then no sound attenuation (ATTN_NONE)
local float attn_value;
attn_value=ATTN_NORM;
if (Goal.option == 1)
attn_value=ATTN_NONE;
// Make the sound
if (Goal.noise)
sound (other, CHAN_ITEM, Goal.noise, 1, attn_value);
winners = FALSE;
// Increase scores
if (Goal.increase_team1 != 0)
{
TeamFortress_TeamIncreaseScore(1, Goal.increase_team1);
//if (chris)
// RoundStop(1);
winners = TRUE;
}
if (Goal.increase_team2 != 0)
{
TeamFortress_TeamIncreaseScore(2, Goal.increase_team2);
//if (chris)
// RoundStop(2);
winners = TRUE;
}
if (Goal.increase_team3 != 0)
{
TeamFortress_TeamIncreaseScore(3, Goal.increase_team3);
//if (chris)
// RoundStop(3);
winners = TRUE;
}
if (Goal.increase_team4 != 0)
{
TeamFortress_TeamIncreaseScore(4, Goal.increase_team4);
//if (chris)
// RoundStop(4);
winners = TRUE;
}
// Display short team scores only if scores changed
if (winners == TRUE)
TeamFortress_TeamShowScores(2);
// CTF Map support
if (CTF_Map == TRUE)
{
if (AP)
{
if (Goal.goal_no == CTF_FLAG1)
{
te = find(NIL, classname, "player");
while (te)
{
if (te.team_no == 2)
{
if (te == AP)
{
winners = random();
if (winners < 0.1)
CenterPrint2(te, "\n\n\n", MSG_CTF_FLAG_GRAB_TEAM1);
else if (winners < 0.2)
CenterPrint2(te, "\n\n\n", MSG_CTF_FLAG_GRAB_TEAM2);
else if (winners < 0.6)
CenterPrint2(te, "\n\n\n", MSG_CTF_FLAG_GRAB_TEAM3);
else if (winners < 0.7)
CenterPrint2(te, "\n\n\n", MSG_CTF_FLAG_GRAB_TEAM4);
else if (winners < 0.8)
CenterPrint2(te, "\n\n\n", MSG_CTF_FLAG_GRAB_TEAM5_2);
else if (winners < 0.95)
CenterPrint2(te, "\n\n\n", MSG_CTF_FLAG_GRAB_TEAM6);
else
CenterPrint2(te, "\n\n\n", MSG_CTF_FLAG_GRAB_TEAM7);
}
else
CenterPrint2(te, "\n\n\n", "Your team <20><><EFBFBD> the <20><><EFBFBD><EFBFBD><EFBFBD> flag!!");
}
else
CenterPrint2(te, "\n\n\n", "Your flag has been <20><><EFBFBD><EFBFBD><EFBFBD>!!");
te = find(te, classname, "player");
}
bprint(PRINT_HIGH, AP.netname);
bprint(PRINT_HIGH, " <20><><EFBFBD> the <20><><EFBFBD><EFBFBD> flag!\n");
AP.items = AP.items | IT_KEY1;
}
else if (Goal.goal_no == CTF_FLAG2)
{
te = find(NIL, classname, "player");
while (te)
{
if (te.team_no == 1)
{
if (te == AP)
{
winners = random();
if (winners < 0.1)
CenterPrint2(te, "\n\n\n", MSG_CTF_FLAG_GRAB_TEAM1);
else if (winners < 0.2)
CenterPrint2(te, "\n\n\n", MSG_CTF_FLAG_GRAB_TEAM2);
else if (winners < 0.6)
CenterPrint2(te, "\n\n\n", MSG_CTF_FLAG_GRAB_TEAM3);
else if (winners < 0.7)
CenterPrint2(te, "\n\n\n", MSG_CTF_FLAG_GRAB_TEAM4);
else if (winners < 0.8)
CenterPrint2(te, "\n\n\n", MSG_CTF_FLAG_GRAB_TEAM5_1);
else if (winners < 0.95)
CenterPrint2(te, "\n\n\n", MSG_CTF_FLAG_GRAB_TEAM6);
else
CenterPrint2(te, "\n\n\n", MSG_CTF_FLAG_GRAB_TEAM7);
}
else
CenterPrint2(te, "\n\n\n", "Your team <20><><EFBFBD> the <20><><EFBFBD><EFBFBD><EFBFBD> flag!!");
}
else
CenterPrint2(te, "\n\n\n", "Your flag has been <20><><EFBFBD><EFBFBD><EFBFBD>!!");
te = find(te, classname, "player");
}
bprint(PRINT_HIGH, AP.netname);
bprint(PRINT_HIGH, " <20><><EFBFBD> the <20><><EFBFBD> flag!\n");
AP.items = AP.items | IT_KEY2;
}
else if (Goal.goal_no == CTF_DROPOFF1)
{
te = find(NIL, classname, "player");
while (te)
{
if (te.team_no == 2)
{
if (te == AP)
CenterPrint2(te, "\n\n\n", "You <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> the flag!!");
else
CenterPrint2(te, "\n\n\n", "Your flag was <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>!!");
}
else
CenterPrint2(te, "\n\n\n", "Your team <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> the flag!!");
te = find(te, classname, "player");
}
bprint(PRINT_HIGH, AP.netname);
bprint(PRINT_HIGH, " <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> the <20><><EFBFBD> flag!\n");
AP.items = AP.items & ~IT_KEY2;
}
else if (Goal.goal_no == CTF_DROPOFF2)
{
te = find(NIL, classname, "player");
while (te)
{
if (te.team_no == 1)
{
if (te == AP)
CenterPrint2(te, "\n\n\n", "You <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> the flag!!");
else
CenterPrint2(te, "\n\n\n", "Your flag was <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>!!");
}
else
CenterPrint2(te, "\n\n\n", "Your team <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> the flag!!");
te = find(te, classname, "player");
}
bprint(PRINT_HIGH, AP.netname);
bprint(PRINT_HIGH, " <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> the <20><><EFBFBD><EFBFBD> flag!\n");
AP.items = AP.items & ~IT_KEY1;
}
}
}
// Go through all the players and do any results
te = find(NIL, classname, "player");
while (te)
{
// Centerprinting
if (Goal.broadcast && CTF_Map == FALSE)
CenterPrint2(te, "\n\n\n", Goal.broadcast);
if (Goal.netname_broadcast && CTF_Map == FALSE)
{
sprint(te, PRINT_HIGH, AP.netname);
sprint(te, PRINT_HIGH, Goal.netname_broadcast);
}
if (AP == te)
{
if (Goal.message)
CenterPrint2(te, "\n\n\n", Goal.message);
}
else if (AP.team_no == te.team_no)
{
if (Goal.owners_team_broadcast && te.team_no == Goal.owned_by)
CenterPrint2(te, "\n\n\n", Goal.owners_team_broadcast);
else if (Goal.team_broadcast)
CenterPrint2(te, "\n\n\n", Goal.team_broadcast);
if (Goal.netname_owners_team_broadcast && te.team_no == Goal.owned_by)
{
sprint(te, PRINT_HIGH, AP.netname);
sprint(te, PRINT_HIGH, Goal.netname_owners_team_broadcast);
}
else if (Goal.netname_team_broadcast)
{
sprint(te, PRINT_HIGH, AP.netname);
sprint(te, PRINT_HIGH, Goal.netname_team_broadcast);
}
}
else
{
if (Goal.owners_team_broadcast && te.team_no == Goal.owned_by)
CenterPrint2(te, "\n\n\n", Goal.owners_team_broadcast);
else if (Goal.non_team_broadcast)
CenterPrint2(te, "\n\n\n", Goal.non_team_broadcast);
if (Goal.netname_owners_team_broadcast && te.team_no == Goal.owned_by)
{
sprint(te, PRINT_HIGH, AP.netname);
sprint(te, PRINT_HIGH, Goal.netname_owners_team_broadcast);
}
else if (Goal.netname_non_team_broadcast)
{
sprint(te, PRINT_HIGH, AP.netname);
sprint(te, PRINT_HIGH, Goal.netname_non_team_broadcast);
}
}
if (IsAffectedBy(Goal, te, AP))
{
// If its a Timer Goal, see if it needs to check Criteria again
if (Goal.search_time != 0 && Goal.goal_effects & TFGE_TIMER_CHECK_AP)
{
if (APMeetsCriteria(Goal, te))
Apply_Results(Goal, te, AP, addb);
}
else
{
Apply_Results(Goal, te, AP, addb);
}
}
te = find(te, classname, "player");
}
// Goal is now active
// Items are not always set to active. They handle their modes.
if (Goal.classname != "item_tfgoal")
Goal.goal_state = TFGS_ACTIVE;
// EndGame checking
if (Goal.goal_result & TFGR_ENDGAME)
{
// Display Long TeamScores to everyone
TeamFortress_TeamShowScores(1);
winners = TeamFortress_TeamGetWinner();
// Stop everyone
te = find (NIL, classname, "player");
while (te)
{
te.takedamage = 0;
te.movetype = MOVETYPE_NONE;
te.velocity = '0 0 0';
te.avelocity = '0 0 0';
te = find(te, classname, "player");
}
te = spawn();
te.nextthink = time + 5; // Allow 5 secs to read the scores
te.think = execute_changelevel;
dremove(Goal);
return;
}
//CH spawn explosions if so needed
if (Goal.goal_effects & TFGE_CAUSE_EXPLOSION)
{
WriteByte (MSG_MULTICAST, SVC_TEMPENTITY);
WriteByte (MSG_MULTICAST, TE_EXPLOSION);
WriteCoord (MSG_MULTICAST, self.origin_x);
WriteCoord (MSG_MULTICAST, self.origin_y);
WriteCoord (MSG_MULTICAST, self.origin_z);
multicast (self.origin, MULTICAST_PHS);
// BecomeExplosion();
}
#ifdef MAP_DEBUG
#ifdef VERBOSE
RPrint("Doing Groupwork...\n");
#endif
#endif
// Do Goal Group checking
DoGroupWork(Goal, AP);
#ifdef MAP_DEBUG
#ifdef VERBOSE
RPrint("Doing Goalwork...\n");
#endif
#endif
// Do Goal checking
DoGoalWork(Goal, AP);
#ifdef MAP_DEBUG
#ifdef VERBOSE
RPrint("Doing Triggerwork...\n");
#endif
#endif
// Do Quake Trigger actions
DoTriggerWork(Goal, AP);
#ifdef MAP_DEBUG
#ifdef VERBOSE
RPrint("Setting up Respawn...\n");
#endif
#endif
if (Goal.killtarget && Goal.killtarget == Goal.targetname) {
// self destructive trigger
remove (Goal);
} else {
// Setup for Respawn
// Items and Triggers do their own respawn work
if (Goal.classname == "info_tfgoal")
SetupRespawn(Goal);
}
};
//=========================================================================
// GOAL FUNCTIONS
//=========================================================================
// Touch function for Goals
void() tfgoal_touch =
{
local entity te;
// If it is not activated in by the player's touch, return
if (!(self.goal_activation & TFGA_TOUCH))
return;
if (other.classname != "player")
return;
// If we're in prematch mode, ignore the touch
if (prematch >= time)
return;
if (other.penance_time >= time)
return;
// If it's already active, don't bother
if (self.goal_state == TFGS_ACTIVE)
{
#ifdef MAP_DEBUG
RPrint("Goal already active. aborting touch.\n");
#endif
return;
}
// CTF Hack to make sure the key is in place.
// Temporary hack :)
if (CTF_Map == TRUE)
{
if ((self.goal_no == CTF_DROPOFF1) && (other.team_no == 1))
{
te = Finditem(CTF_FLAG1);
if ((te.goal_state == TFGS_ACTIVE) || (te.origin != te.oldorigin))
return;
}
if ((self.goal_no == CTF_DROPOFF2) && (other.team_no == 2))
{
te = Finditem(CTF_FLAG2);
if ((te.goal_state == TFGS_ACTIVE) || (te.origin != te.oldorigin))
return;
}
}
AttemptToActivate(self, other, self);
};
//=========================================================================
// Use (Triggered) function for Goals
void() info_tfgoal_use =
{
#ifdef MAP_DEBUG
RPrint("\n");
RPrint("TFGoal triggered by other entity.\n");
#endif
AttemptToActivate(self, activator, self);
};
//=========================================================================
// Timer goal tick
void() tfgoal_timer_tick =
{
// Check criteria
if (self.goal_state != TFGS_REMOVED)
{
#ifdef MAP_DEBUG
RPrint("==========================\n");
#ifdef VERBOSE
RPrint("Timer Tick for: ");
#else
RPrint("Tick: ");
#endif
RPrint(self.netname);
RPrint(" Checking criteria...");
#endif
if (APMeetsCriteria(self, NIL))
{
DoResults(self, NIL, TRUE);
}
else
{
InactivateGoal(self);
self.think = tfgoal_timer_tick;
self.nextthink = time + self.search_time;
}
}
};
//=========================================================================
// GOALITEM FUNCTIONS
//=========================================================================
// Touch function for the goalitem entity
void() item_tfgoal_touch =
{
if (infokey(NIL,"ceasefire")=="on") //OfN
return;
local entity te;
if (other.classname != "player")
return;
if (other.health <= 0)
return;
// Don't let them take goalitems in prematch
if (prematch >= time)
return;
// Don't let lame teamkillers take items
if (other.penance_time >= time)
return;
// CTF Hack to return your key.
// Temporary hack :)
if (CTF_Map == TRUE)
{
if (self.goal_no == CTF_FLAG1)
{
// Flag not at home?
if (self.origin != self.oldorigin)
{
if (other.team_no == 1)
{
bprint(PRINT_HIGH, other.netname);
bprint(PRINT_HIGH, " <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> the <20><><EFBFBD><EFBFBD> flag!\n");
te = find(NIL, classname, "player");
while (te)
{
if (te.team_no == 1)
CenterPrint2(te, "\n\n\n", "Your flag was <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>!!");
else
CenterPrint2(te, "\n\n\n", "The <20><><EFBFBD><EFBFBD><EFBFBD> flag was <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>!!");
te = find(te, classname, "player");
}
self.goal_state = TFGS_INACTIVE;
self.solid = SOLID_TRIGGER;
self.touch = item_tfgoal_touch;
self.origin = self.oldorigin;
setmodel(self, self.mdl);
setorigin(self, self.origin);
sound (self, CHAN_VOICE, "items/itembk2.wav", 1, ATTN_NORM);
return;
}
}
else
{
if (other.team_no == 1)
return;
}
}
else if (self.goal_no == CTF_FLAG2)
{
// Flag not at home?
if (self.origin != self.oldorigin)
{
if (other.team_no == 2)
{
bprint(PRINT_HIGH, other.netname);
bprint(PRINT_HIGH, " <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> the <20><><EFBFBD> flag!\n");
te = find(NIL, classname, "player");
while (te)
{
if (te.team_no == 2)
CenterPrint(te, "\n\n\n Your flag was <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>!!");
else
CenterPrint(te, "\n\n\n The <20><><EFBFBD><EFBFBD><EFBFBD> flag was <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>!!");
te = find(te, classname, "player");
}
self.goal_state = TFGS_INACTIVE;
self.solid = SOLID_TRIGGER;
self.touch = item_tfgoal_touch;
self.origin = self.oldorigin;
setmodel(self, self.mdl);
setorigin(self, self.origin);
sound (self, CHAN_VOICE, "items/itembk2.wav", 1, ATTN_NORM);
return;
}
}
else
{
if (other.team_no == 2)
return;
}
}
}
if (Activated(self, other))
{
// Give it to the player
tfgoalitem_GiveToPlayer(self, other, self);
self.goal_state = TFGS_ACTIVE;
}
else
{
// If an else goal should be activated, activate it
if (self.else_goal != 0)
{
#ifdef MAP_DEBUG
#ifdef VERBOSE
RPrint(" Else Goal.\n");
#endif
#endif
te = Findgoal(self.else_goal);
if (te)
AttemptToActivate(te, other, self);
}
}
};
//=========================================================================
// Give the GoalItem to a Player.
void(entity Item, entity AP, entity Goal) tfgoalitem_GiveToPlayer =
{
#ifdef MAP_DEBUG
RPrint("Giving ");
RPrint(Item.netname);
RPrint(" to ");
RPrint(AP.netname);
RPrint("\n");
#endif
Item.owner = AP;
// Remove it from the map
if (Item.mdl)
setmodel(Item, "");
Item.solid = SOLID_NOT;
// Do the deeds on the player
if (Item.goal_activation & TFGI_GLOW)
AP.effects = AP.effects | EF_GlowColor(AP);
if (Item.goal_activation & TFGI_SLOW)
TeamFortress_SetSpeed(AP);
if (Item.goal_activation & TFGI_ITEMGLOWS)
Item.effects &= ~EF_ANYGLOW;
// Light up console icons
if (Item.items & IT_KEY1)
AP.items = AP.items | IT_KEY1;
if (Item.items & IT_KEY2)
AP.items = AP.items | IT_KEY2;
// Only do the results if we're allowed to
if (Goal != Item)
{
if (Goal.goal_result & TFGR_NO_ITEM_RESULTS)
{
Item.goal_state = TFGS_ACTIVE;
return;
}
}
#ifdef MAP_DEBUG
#ifdef VERBOSE
RPrint("Doing item results...\n");
#endif
#endif
// Prevent the Player from disguising themself if applicable
//CH added looking for spykit & thief
if (Item.goal_result & TFGR_REMOVE_DISGUISE)
{
if (AP.job & JOB_THIEF && (AP.job & JOB_ACTIVE || AP.job & JOB_FULL_HIDE))
RevealThief(AP,FALSE);
if (AP.playerclass == PC_SPY || AP.cutf_items & CUTF_SPY_KIT)
{
AP.is_unabletospy = 1;
}
}
// Do the Results, adding the bonuses
DoResults(Item, AP, TRUE);
// Check the Item Group Stuff
DoItemGroupWork(Item, AP);
};
//=========================================================================
//Update which team has the flag
void(float item) Update_team_with_flag_touch =
{
if ( mapname != "steal4d" )
{
team_with_flag = 0;
return;
}
if (item == 103)
{
team_with_flag = 1;
friends1_mask = 0;
friends2_mask = friends3_mask = friends4_mask = 14;
return;
}
if (item == 203) {
team_with_flag = 2;
friends2_mask = 0;
friends1_mask = friends3_mask = friends4_mask = 13;
return;
}
if (item == 303) {
team_with_flag = 3;
friends3_mask = 0;
friends1_mask = friends2_mask = friends4_mask = 11;
return;
}
if (item == 403) {
team_with_flag = 4;
friends4_mask = 0;
friends1_mask = friends2_mask = friends3_mask = 7;
return;
}
};
//=========================================================================
//Update which team has the flag
void(float item) Update_team_with_flag_drop =
{
if ( mapname != "steal4d" ){
team_with_flag = 0;
return;
}
if (item != 103 && item != 203 && item != 303 && item != 403)
return;
team_with_flag= 0;
friends1_mask= friends2_mask= friends3_mask= friends4_mask= 0;
};
//=========================================================================
// Return the Item to its starting point
void() ReturnItem =
{
self.enemy.goal_state = TFGS_INACTIVE;
//CH sets solid type
if (self.enemy.goal_activation & TFGI_GOAL_IS_SOLID)
self.enemy.solid = SOLID_BBOX;
else
self.enemy.solid = SOLID_TRIGGER;
self.enemy.movetype = MOVETYPE_NONE;
self.enemy.touch = item_tfgoal_touch;
self.enemy.origin = self.enemy.oldorigin;
if (self.enemy.mdl)
setmodel(self.enemy, self.enemy.mdl);
setorigin(self.enemy, self.enemy.origin);
sound(self.enemy, CHAN_VOICE, "items/itembk2.wav", 1, ATTN_NORM);
// Restore a goal if needed
tfgoalitem_checkgoalreturn(self.enemy);
dremove(self);
};
//=========================================================================
// Remove the GoalItem from a Player.
void(entity Item, entity AP, float method) tfgoalitem_RemoveFromPlayer =
{
local entity te;
local float slowon;
local float key1on, key2on;
local float spyoff;
local entity DelayReturn;
if (!Item)
{
RPrint("error: tfgoalitem_RemoveFromPlayer(): !Item");
return;
}
#ifdef MAP_DEBUG
RPrint("Removing ");
RPrint(Item.netname);
RPrint(" from ");
RPrint(AP.netname);
RPrint("\n");
#endif
slowon = FALSE;
key1on = FALSE;
key2on = FALSE;
spyoff = FALSE;
// Remove the deeds from the player
// Make sure we don't remove an effect another Goal is also supplying
AP.effects &= ~EF_ANYGLOW;
if (Item.goal_activation & TFGI_ITEMGLOWS)
Item.effects = Item.effects | EF_GlowColor(Item);
// Remove the Spy prevention
if (!spyoff)
AP.is_unabletospy = 0;
// Remove the Light up of console icons
if (!key1on)
AP.items = AP.items & ~IT_KEY1;
if (!key2on)
AP.items = AP.items & ~IT_KEY2;
// Remove AP Modifications
// Go through all the players and do any results
te = find(NIL, classname, "player");
while (te)
{
if (IsAffectedBy(Item, te, AP))
{
RemoveResults(Item, te);
}
te = find(te, classname, "player");
}
/*
db1 = ftos(method);
RPrint("returning via method : ");
RPrint(db1);
RPrint("\n");
*/
// Return it to the starting point if the flag is set
if (method == 0) // Dropped by a dying player
{
// Do messages
te = find(NIL, classname, "player");
while (te)
{
if (te.team_no == Item.owned_by)
{
if (Item.team_drop)
CenterPrint2(te, "\n\n\n", Item.team_drop);
if (Item.netname_team_drop)
{
sprint(te, PRINT_HIGH, AP.netname);
sprint(te, PRINT_HIGH, Item.netname_team_drop);
}
}
else // (te.team_no != Item.owned_by)
{
if (Item.non_team_drop)
CenterPrint2(te, "\n\n\n", Item.non_team_drop);
if (Item.netname_non_team_drop)
{
sprint(te, PRINT_HIGH, AP.netname);
sprint(te, PRINT_HIGH, Item.netname_non_team_drop);
}
}
te = find(te, classname, "player");
}
// Drop it if the flag is set
if (Item.goal_activation & TFGI_RETURN_DROP)
{
DelayReturn = spawn();
DelayReturn.enemy = Item;
DelayReturn.think = ReturnItem;
DelayReturn.nextthink = time + 0.5;
/*
Item.solid = SOLID_TRIGGER;
Item.touch = item_tfgoal_touch;
Item.origin = Item.oldorigin;
setmodel(Item, Item.mdl);
setorigin(Item, Item.origin);
sound (Item, CHAN_VOICE, "items/itembk2.wav", 1, ATTN_NORM);
*/
}
else if (Item.goal_activation & TFGI_DROP)
{
tfgoalitem_drop(Item);
}
else
{
// Remove the Item
Item.owner = NIL;
dremove(Item);
TeamFortress_SetSpeed(AP);
AP.effects |= EF_GlowColor(AP);
return;
}
Item.owner = NIL;
TeamFortress_SetSpeed(AP);
AP.effects |= EF_GlowColor(AP);
return;
}
// Return it to the starting point if the flag is set
if (method == 1) // Removed by a goal activation
{
/*
bprint(Item.netname);
bprint(" ");
bprint(Item.mdl);
bprint(" ");
bprint(vtos(Item.origin));
bprint(" ");
bprint(vtos(Item.oldorigin));
bprint("\n");
*/
if (Item.goal_activation & TFGI_RETURN_GOAL)
{
DelayReturn = spawn();
DelayReturn.enemy = Item;
DelayReturn.think = ReturnItem;
DelayReturn.nextthink = time + 0.5;
/*
Item.origin = Item.oldorigin;
setorigin(Item, Item.origin);
Item.solid = SOLID_TRIGGER;
Item.touch = item_tfgoal_touch;
setmodel(Item, Item.mdl);
sound (Item, CHAN_VOICE, "items/itembk2.wav", 1, ATTN_NORM);
// Restore a goal if needed
tfgoalitem_checkgoalreturn(Item);
*/
Item.owner = NIL;
TeamFortress_SetSpeed(AP);
AP.effects |= EF_GlowColor(AP);
return;
}
// Don't remove it, since it may be given away again later
Item.solid = SOLID_NOT;
Item.owner = NIL;
TeamFortress_SetSpeed(AP);
AP.effects |= EF_GlowColor(AP);
return;
}
RPrint("Invalid method passed into tfgoalitem_RemoveFromPlayer\n");
};
//=========================================================================
// A quick check to make sure the items is not in a wall
void() tfgoalitem_dropthink =
{
local float pos;
#ifdef MAP_DEBUG
RPrint("DropThink for ");
RPrint(self.netname);
RPrint("\n");
#endif
// If the flag is set, remove it after 2 minutes
self.movetype = MOVETYPE_TOSS;
if (self.pausetime != 0)
{
pos = pointcontents(self.origin); // find the location of the Item
if (pos == CONTENTS_SLIME)
self.nextthink = time + (self.pausetime / 4);
else if (pos == CONTENTS_LAVA)
self.nextthink = time + 5;
else if (pos == CONTENTS_SOLID || pos == CONTENTS_SKY) // oh shit!
self.nextthink = time + 2;
else
self.nextthink = time + self.pausetime;
self.think = tfgoalitem_remove;
}
};
//CH does same as above but sets origion when done.
void() tfgoalitem_dropthink2 =
{
local float pos;
#ifdef MAP_DEBUG
RPrint("DropThink for ");
RPrint(self.netname);
RPrint("\n");
#endif
// If the flag is set, remove it after 2 minutes
self.movetype = MOVETYPE_TOSS;
/* CH leave this as it is working code. that below this is a test, which seems to work well.
// if (self.pausetime != 0)
// {
pos = pointcontents(self.origin); // find the location of the Item
if (pos == CONTENTS_SLIME)
self.nextthink = time + (self.pausetime / 4);
else if (pos == CONTENTS_LAVA)
self.nextthink = time + 5;
else if (pos == CONTENTS_SOLID || pos == CONTENTS_SKY) // oh shit!
self.nextthink = time + 2;
else
self.nextthink = time + self.pausetime;
self.think = tfgoalitem_remove;
self.velocity = '0 0 0';
self.oldorigin = self.origin;
// }
*/
// if (self.pausetime != 0)
// {
pos = pointcontents(self.origin); // find the location of the Item
if (pos == CONTENTS_SLIME)
setorigin(self, self.oldorigin);
else if (pos == CONTENTS_LAVA)
setorigin(self, self.oldorigin);
else if (pos == CONTENTS_SOLID || pos == CONTENTS_SKY) // oh shit!
setorigin(self, self.oldorigin);
else
setorigin(self, self.origin);
self.velocity = '0 0 0';
self.oldorigin = self.origin;
// }
};
//=========================================================================
// Drop the item just like a backpack
void(entity Item) tfgoalitem_drop =
{
local vector temp1;
local vector temp2;
Item.origin = Item.owner.origin;
Item.velocity_z = 400;
Item.velocity_x = -50 + (random() * 100);
Item.velocity_y = -50 + (random() * 100);
Item.movetype = MOVETYPE_TOSS;
//CH set solid state
//GR this can cause players to be trapped inside their items when they dropitems, do it later
#if 0
if (Item.goal_activation & TFGI_GOAL_IS_SOLID)
Item.solid = SOLID_BBOX;
else
#endif
Item.solid = SOLID_TRIGGER;
setorigin(Item, Item.origin);
//CH sets goal bounding box size
if (Item.goal_min)
temp1 = Item.goal_min;
else
temp1 = '-16 -16 -24';
if (Item.goal_max)
temp2 = Item.goal_max;
else
temp2 = '16 16 32';
setsize (Item, temp1, temp2); //CH sets box size from above
if (Item.mdl)
setmodel(Item, Item.mdl);
if (!Item.owner.deadflag) //checks for dropitems
{
makevectors(Item.owner.v_angle);
Item.velocity = v_forward * 400 + v_up * 200;
Item.nextthink = time + 0.5; //CH wait a sec then make it pickupable GR half a sec
Item.think = tfgoalitem_settouchandthink;
}
else
{
Item.nextthink = time + 5.0; // give it five seconds
Item.think = tfgoalitem_dropthink; // and then find where it ended up
Item.goal_state = TFGS_INACTIVE;
Item.touch = item_tfgoal_touch;
if (Item.goal_activation & TFGI_GOAL_IS_SOLID)
{
Item.solid = SOLID_BBOX;
setorigin(Item, Item.origin);
}
}
};
//CH brings the item online after 1 sec
void() tfgoalitem_settouchandthink =
{
self.nextthink = time + 4.0; // give it five seconds
self.think = tfgoalitem_dropthink; // and then find where it ended up
self.goal_state = TFGS_INACTIVE;
self.touch = item_tfgoal_touch;
if (self.goal_activation & TFGI_GOAL_IS_SOLID)
{
self.solid = SOLID_BBOX;
setorigin(self, self.origin);
}
};
//=========================================================================
// Remove the item, or Return it if needed
void() tfgoalitem_remove =
{
local entity te;
#ifdef MAP_DEBUG
RPrint("RemoveItem for ");
RPrint(self.netname);
RPrint("... ");
#endif
// Has someone picked it up?
if (self.goal_state == TFGS_ACTIVE)
return;
// Should it be returned?
if (self.goal_activation & TFGI_RETURN_REMOVE)
{
#ifdef MAP_DEBUG
RPrint("Returned.\n");
#endif
//CH set solid state
if (self.goal_activation & TFGI_GOAL_IS_SOLID)
self.solid = SOLID_BBOX;
else
self.solid = SOLID_TRIGGER;
self.movetype = MOVETYPE_NONE;
self.touch = item_tfgoal_touch;
self.origin = self.oldorigin;
if (self.mdl)
setmodel(self, self.mdl);
setorigin (self, self.origin);
sound (self, CHAN_VOICE, "items/itembk2.wav", 1, ATTN_NORM);
// Restore a goal if needed
tfgoalitem_checkgoalreturn(self);
//CH could it check origin again?? nah...
//CH a better question would be why does it need this to work right???
self.think = tfgoalitem_checkoriginagain;
self.nextthink = time + 1;
// Do we need to do any CenterPrint2ing?
if (self.noise3 || self.noise4)
{
te = find (NIL, classname, "player");
while (te)
{
if (te.team_no == self.owned_by)
CenterPrint2(te, "\n\n\n", self.noise3);
else // (te.team_no != self.owned_by)
CenterPrint2(te, "\n\n\n", self.noise4);
te = find(te, classname, "player");
}
}
return;
}
#ifdef MAP_DEBUG
RPrint("Removed.\n");
#endif
dremove(self);
};
//===================
//CH i wonder what this does?
//CH WHY THE HELL DOES IT NEED THIS!!!!!!!
//CH first check
void() tfgoalitem_checkoriginagain =
{
// RPrint("Check #1 for return goalitem\n");
if (self.origin != self.oldorigin && self.goal_state != TFGS_ACTIVE)
{
self.origin = self.oldorigin;
setorigin (self, self.origin);
self.think = tfgoalitem_checkoriginagain2;
self.nextthink = time + 1;
}
};
//===================
//CH second check
void() tfgoalitem_checkoriginagain2 =
{
// RPrint("Check #2 for return goalitem\n");
if (self.origin != self.oldorigin && self.goal_state != TFGS_ACTIVE)
{
self.origin = self.oldorigin;
setorigin (self, self.origin);
self.think = tfgoalitem_checkoriginagain3;
self.nextthink = time + 1;
}
};
//===================
//CH third and final check
void() tfgoalitem_checkoriginagain3 =
{
// RPrint("Check #3 for return goalitem\n");
if (self.origin != self.oldorigin && self.goal_state != TFGS_ACTIVE)
{
self.origin = self.oldorigin;
setorigin (self, self.origin);
}
};
//=========================================================================
// Activate a goal when this item is removed, if needed
void(entity Item) tfgoalitem_checkgoalreturn =
{
local entity te;
// Do we need to activate a goal somewhere?
if (Item.impulse != 0)
{
// Find the goal
te = Findgoal(Item.impulse);
if (te)
{
te = Findgoal(Item.impulse);
if (te)
AttemptToActivate(te, NIL, Item);
}
}
};
//=========================================================================
// Displays the state of a GoalItem
void(entity Goal, entity Player, entity Item) DisplayItemStatus =
{
if (Item.goal_state == TFGS_ACTIVE)
{
if (Player.team_no == Item.owned_by)
sprint(Player, PRINT_HIGH, Goal.team_str_carried);
else
sprint(Player, PRINT_HIGH, Goal.non_team_str_carried);
sprint(Player, PRINT_HIGH, " ");
if (Player == Item.owner)
sprint(Player, PRINT_HIGH, " You");
else
sprint(Player, PRINT_HIGH, Item.owner.netname);
sprint(Player, PRINT_HIGH, ".");
}
else if (Item.origin != Item.oldorigin)
{
if (Player.team_no == Item.owned_by)
sprint(Player, PRINT_HIGH, Goal.team_str_moved);
else
sprint(Player, PRINT_HIGH, Goal.non_team_str_moved);
}
else
{
if (Player.team_no == Item.owned_by)
sprint(Player, PRINT_HIGH, Goal.team_str_home);
else
sprint(Player, PRINT_HIGH, Goal.non_team_str_home);
}
/*CH prints out the origin and oldorigin via flaginfo
sprint(Player, PRINT_HIGH, "\n");
ac = vtos(Item.origin);
sprint(Player, PRINT_HIGH, ac);
sprint(Player, PRINT_HIGH, " ");
ac = vtos(Item.oldorigin);
sprint(Player, PRINT_HIGH, ac);
*/
sprint(Player, PRINT_HIGH, "\n");
};
//=========================================================================
// CTF SUPPORT
//=========================================================================
// Spawn Functions for CTF entities.
// Team 1 spawnpoints
void() info_player_team1 =
{
CTF_Map = TRUE;
// Convert to a TeamFortress TeamSpawn point
self.classname = "info_player_teamspawn";
// We swap them, so that the colors match ours
self.team_no = 2;
// Make this point remove itself after being spawned on
self.goal_effects = TFSP_REMOVESELF;
self.team_str_home = "ts2";
};
// Team 2 spawnpoints
void() info_player_team2 =
{
CTF_Map = TRUE;
// Convert to a TeamFortress TeamSpawn point
self.classname = "info_player_teamspawn";
// We swap them, so that the colors match ours
self.team_no = 1;
// Make this point remove itself after being spawned on
self.goal_effects = TFSP_REMOVESELF;
self.team_str_home = "ts1";
};
// Team 2 flag
void() item_flag_team2 =
{
local entity dp;
CTF_Map = TRUE;
precache_model ("progs/w_s_key.mdl");
precache_sound ("ogre/ogwake.wav");
precache_sound ("boss2/pop2.wav");
// Convert the flag to a TeamFortress Goal Item
self.classname = "item_tfgoal";
self.netname = "Team 1 Flag";
self.broadcast = " <20><><EFBFBD> the enemy team's flag!\n";
self.deathtype = "You've got the enemy flag!\n";
self.noise = "ogre/ogwake.wav";
self.mdl = "progs/tf_flag.mdl";
self.skin = 0;
setmodel (self, self.mdl);
self.goal_no = CTF_FLAG1;
self.goal_activation = TFGI_GLOW | TFGI_DROP | TFGI_REMOVE | TFGI_RETURN_REMOVE | TFGI_RETURN_GOAL | TFGI_ITEMGLOWS;
self.goal_effects = 1;
self.pausetime = 128;
setsize (self, '-16 -16 -24', '16 16 32');
self.touch = item_tfgoal_touch;
self.goal_state = TFGS_INACTIVE;
self.solid = SOLID_TRIGGER;
setorigin(self, self.origin);
self.nextthink = time + 0.2; // items start after other solids
self.think = TF_PlaceItem;
// Create the dropoff point Goal
dp = spawn();
dp.origin = self.origin;
dp.classname = "info_tfgoal";
dp.goal_activation = 1;
dp.team_no = 1;
dp.items_allowed = CTF_FLAG2;
dp.goal_no = CTF_DROPOFF1;
dp.goal_effects = 3;
dp.broadcast = " <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> the enemy flag!\n";
dp.message = "You <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> the enemy flag!\n";
dp.noise = "boss2/pop2.wav";
dp.goal_result = TFGR_ADD_BONUSES;
dp.activate_goal_no = CTF_SCORE1;
dp.axhitme = CTF_FLAG2;
dp.count = 10;
dp.frags = 0;
dp.solid = SOLID_TRIGGER;
dp.goal_state = TFGS_INACTIVE;
setsize (dp, '-16 -16 -24', '16 16 32');
dp.nextthink = time + 0.2; // goals start after other solids
dp.think = TF_PlaceGoal;
// Create the extra scoring goal
dp = spawn();
dp.origin = dp.origin;
dp.classname = "info_tfgoal";
dp.goal_effects = 1;
dp.frags = 5;
dp.goal_activation = 0;
dp.goal_no = CTF_SCORE1;
dp.solid = SOLID_NOT;
dp.goal_state = TFGS_INACTIVE;
setsize (dp, '-16 -16 -24', '16 16 32');
dp.nextthink = time + 0.2; // goals start after other solids
dp.think = TF_PlaceGoal;
};
// Team 1 flag
void() item_flag_team1 =
{
local entity dp;
CTF_Map = TRUE;
precache_model ("progs/tf_flag.mdl");
precache_sound ("ogre/ogwake.wav");
precache_sound ("boss2/pop2.wav");
// Convert the flag to a TeamFortress Goal Item
self.classname = "item_tfgoal";
self.netname = "Team 2 Flag";
self.broadcast = " <20><><EFBFBD> the enemy team's flag!\n";
self.deathtype = "You've got the enemy flag!\n";
self.noise = "ogre/ogwake.wav";
self.mdl = "progs/tf_flag.mdl";
setmodel (self, self.mdl);
self.skin = 1;
self.goal_no = CTF_FLAG2;
self.goal_activation = TFGI_GLOW | TFGI_DROP | TFGI_REMOVE | TFGI_RETURN_REMOVE | TFGI_RETURN_GOAL | TFGI_ITEMGLOWS;
self.goal_effects = 1;
self.pausetime = 128;
setsize (self, '-16 -16 -24', '16 16 32');
self.touch = item_tfgoal_touch;
self.goal_state = TFGS_INACTIVE;
self.solid = SOLID_TRIGGER;
setorigin(self, self.origin);
self.nextthink = time + 0.2; // items start after other solids
self.think = TF_PlaceItem;
// Create the dropoff point Goal
dp = spawn();
dp.origin = self.origin;
dp.classname = "info_tfgoal";
dp.goal_activation = 1;
dp.team_no = 2;
dp.items_allowed = CTF_FLAG1;
dp.goal_no = CTF_DROPOFF2;
dp.goal_effects = 3;
dp.broadcast = " <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> the enemy flag!\n";
dp.message = "You <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> the enemy flag!\n";
dp.noise = "boss2/pop2.wav";
dp.goal_result = TFGR_ADD_BONUSES;
dp.activate_goal_no = CTF_SCORE2;
dp.axhitme = CTF_FLAG1;
dp.count = 10;
dp.frags = 0;
dp.solid = SOLID_TRIGGER;
dp.goal_state = TFGS_INACTIVE;
setsize (dp, '-16 -16 -24', '16 16 32');
dp.nextthink = time + 0.2; // goals start after other solids
dp.think = TF_PlaceGoal;
// Create the extra scoring goal
dp = spawn();
dp.origin = dp.origin;
dp.classname = "info_tfgoal";
dp.goal_effects = 1;
dp.frags = 5;
dp.goal_activation = 0;
dp.goal_no = CTF_SCORE2;
dp.solid = SOLID_NOT;
dp.goal_state = TFGS_INACTIVE;
setsize (dp, '-16 -16 -24', '16 16 32');
dp.nextthink = time + 0.2; // goals start after other solids
dp.think = TF_PlaceGoal;
};
//=========================================================================
// Checks to make sure either flag hasn't got into bad state, and if
// it has, returns it.
void() CTF_FlagCheck =
{
local entity te;
local float flagcount, pos;
flagcount = 0;
te = find(NIL, classname, "item_tfgoal");
while (te)
{
if (te.goal_no == CTF_FLAG1)
{
pos = pointcontents(te.origin); // find the location of the Flag
if (pos == CONTENTS_SOLID || pos == CONTENTS_SKY)
{
RPrint("*****BUG*****\nFlag(s) outside world.\nPlease report this.\n");
te.nextthink = time + 0.2;
te.think = tfgoalitem_remove;
}
flagcount = flagcount + 1;
}
else if (te.goal_no == CTF_FLAG2)
{
pos = pointcontents(te.origin); // find the location of the Flag
if (pos == CONTENTS_SOLID || pos == CONTENTS_SKY)
{
RPrint("*****BUG*****\nFlag(s) outside world.\nPlease report this.\n");
te.nextthink = time + 0.2;
te.think = tfgoalitem_remove;
}
flagcount = flagcount + 1;
}
te = find(te, classname, "item_tfgoal");
}
if (flagcount != 2)
RPrint("*****BUG*****\nFlag(s) missing.\nPlease report this.\n");
self.nextthink = time + 30;
};