mirror of
https://git.code.sf.net/p/quake/quakeforge
synced 2025-01-25 18:21:22 +00:00
738 lines
15 KiB
C++
738 lines
15 KiB
C++
|
|
entity stemp, otemp, s, old;
|
|
|
|
|
|
void() trigger_reactivate =
|
|
{
|
|
self.solid = SOLID_TRIGGER;
|
|
};
|
|
|
|
|
|
float USED_SPAWN_PARMS = 8;
|
|
|
|
void() EncodeLevelParms =
|
|
{
|
|
other.items = other.items - (other.items & (IT_KEY1 | IT_KEY2) );
|
|
if (other.health > 100)
|
|
other.health = 100;
|
|
|
|
parm1 = other.items;
|
|
parm2 = other.health;
|
|
parm3 = other.armorvalue;
|
|
parm4 = other.ammo_shells;
|
|
parm5 = other.ammo_nails;
|
|
parm6 = other.ammo_rockets;
|
|
parm7 = other.weapon;
|
|
parm8 = other.armortype;
|
|
};
|
|
|
|
void() SetNewGameParms =
|
|
{
|
|
other = self;
|
|
|
|
other.health = 100;
|
|
other.ammo_shells = 25;
|
|
other.ammo_nails = 0;
|
|
other.ammo_rockets = 0;
|
|
other.ammo_cells = 0;
|
|
other.items = IT_SHOTGUN | IT_AXE;
|
|
other.weapon = 1;
|
|
other.armortype = 0;
|
|
other.armorvalue = 0;
|
|
|
|
EncodeLevelParms ();
|
|
};
|
|
|
|
void() DecodeLevelParms =
|
|
{
|
|
self.items = parm1;
|
|
self.health = parm2;
|
|
self.armorvalue = parm3;
|
|
self.ammo_shells = parm4;
|
|
self.ammo_nails = parm5;
|
|
self.ammo_rockets = parm6;
|
|
self.weapon = parm7;
|
|
self.armortype = parm8;
|
|
};
|
|
|
|
void() T_changelevel =
|
|
{
|
|
if (other.classname != "player")
|
|
return;
|
|
|
|
self.nextthink = time + 10;
|
|
self.think = trigger_reactivate;
|
|
self.solid = SOLID_NOT;
|
|
|
|
EncodeLevelParms ();
|
|
bprint ("\n\n");
|
|
bprint (other.netname);
|
|
bprint (" killed ");
|
|
bprint (ftos(other.killed_monsters));
|
|
bprint (" monsters out of ");
|
|
bprint (ftos(total_monsters));
|
|
bprint ("\n");
|
|
|
|
bprint ("And found ");
|
|
bprint (ftos(other.found_secrets));
|
|
bprint (" secrets out of ");
|
|
bprint (ftos(total_secrets));
|
|
bprint ("\n\n");
|
|
changelevel (other, self.map, USED_SPAWN_PARMS);
|
|
};
|
|
|
|
/*QUAKED trigger_changelevel (0.5 0.5 0.5) ?
|
|
When the player touches this, he gets sent to the map listed in the "map" variable.
|
|
*/
|
|
void() trigger_changelevel =
|
|
{
|
|
if (!self.map)
|
|
objerror ("chagnelevel trigger doesn't have map");
|
|
|
|
self.angles = '0 0 0';
|
|
self.solid = SOLID_TRIGGER;
|
|
settriggermodel (self, self.model);
|
|
self.touch = T_changelevel;
|
|
};
|
|
|
|
//=============================================================================
|
|
|
|
float SPAWNFLAG_NOMESSAGE = 1;
|
|
float SPAWNFLAG_NOTOUCH = 1;
|
|
|
|
// the wait time has passed, so set back up for another activation
|
|
void() multi_wait =
|
|
{
|
|
if (self.max_health)
|
|
{
|
|
self.health = self.max_health;
|
|
self.takedamage = DAMAGE_YES;
|
|
self.solid = SOLID_BBOX;
|
|
}
|
|
else
|
|
self.solid = SOLID_TRIGGER;
|
|
};
|
|
|
|
|
|
// the delay time has passed, so activate all targets
|
|
void() multi_fire =
|
|
{
|
|
activator = self.enemy;
|
|
|
|
SUB_UseTargets();
|
|
|
|
if (self.wait > 0)
|
|
{
|
|
self.think = multi_wait;
|
|
self.nextthink = time + self.wait;
|
|
}
|
|
else
|
|
self.nextthink = -1;
|
|
};
|
|
|
|
// the trigger was just touched/killed/used
|
|
// self.enemy should be set to the activator so it can be held through a delay
|
|
// so wait for the delay time before firing
|
|
void() multi_trigger =
|
|
{
|
|
if (self.nextthink > time)
|
|
{
|
|
return; // already been triggered
|
|
}
|
|
|
|
if (self.classname == "trigger_secret")
|
|
{
|
|
if (self.enemy.classname != "player")
|
|
return;
|
|
self.enemy.found_secrets = self.enemy.found_secrets + 1;
|
|
WriteByte (self.enemy, SVC_FOUNDSECRET);
|
|
}
|
|
|
|
if (self.message)
|
|
{
|
|
if (self.enemy.classname == "player")
|
|
{
|
|
sound (self.enemy, 0, "temp/talk.wav", 1,1);
|
|
centerprint (self.enemy, self.message);
|
|
}
|
|
}
|
|
|
|
if (self.noise)
|
|
sound (self, 1, self.noise, 1, 1);
|
|
|
|
// don't trigger again until reset
|
|
self.solid = SOLID_NOT;
|
|
self.takedamage = DAMAGE_NO;
|
|
|
|
// either fire now, or after a delay
|
|
if (!self.delay)
|
|
{
|
|
multi_fire ();
|
|
if (self.wait == -1)
|
|
remove(self);
|
|
}
|
|
else
|
|
{
|
|
self.nextthink = time + self.delay;
|
|
self.think = multi_fire;
|
|
}
|
|
|
|
};
|
|
|
|
void() multi_killed =
|
|
{
|
|
self.enemy = damage_attacker;
|
|
multi_trigger();
|
|
};
|
|
|
|
void() multi_use =
|
|
{
|
|
self.enemy = activator;
|
|
multi_trigger();
|
|
};
|
|
|
|
void() multi_touch =
|
|
{
|
|
local vector for;
|
|
|
|
if (other.classname != "player")
|
|
return;
|
|
|
|
// if the trigger has an angles field, check player's facing direction
|
|
if (self.angles != '0 0 0')
|
|
{
|
|
makevectors (self.angles);
|
|
for = v_forward;
|
|
makevectors (other.angles);
|
|
if (v_forward * for < 0)
|
|
return; // not facing the right way
|
|
}
|
|
|
|
self.enemy = other;
|
|
multi_trigger ();
|
|
};
|
|
|
|
|
|
/*QUAKED trigger_multiple (.5 .5 .5) ? notouch
|
|
Variable sized repeatable trigger. Must be targeted at one or more entities. If "health" is set, the trigger must be killed to activate each time.
|
|
If "delay" is set, the trigger waits some time after activating before firing.
|
|
"wait" : Seconds between triggerings. (.2 default)
|
|
If notouch is set, the trigger is only fired by other entities, not by touching.
|
|
sounds
|
|
1) secret
|
|
2) beep beep
|
|
3) large switch
|
|
4)
|
|
set "message" to text string
|
|
*/
|
|
void() trigger_multiple =
|
|
{
|
|
if (self.sounds == 1)
|
|
{
|
|
precache_sound ("temp/secret.wav");
|
|
self.noise = "temp/secret.wav";
|
|
}
|
|
else if (self.sounds == 2)
|
|
{
|
|
precache_sound ("temp/talk.wav");
|
|
self.noise = "temp/talk.wav";
|
|
}
|
|
else if (self.sounds == 3)
|
|
{
|
|
precache_sound ("misc/trigger1.wav");
|
|
self.noise = "misc/trigger1.wav";
|
|
}
|
|
|
|
// self.angles = '0 0 0';
|
|
if (!self.wait)
|
|
self.wait = 0.2;
|
|
self.use = multi_use;
|
|
|
|
if (self.health)
|
|
{
|
|
if (self.spawnflags & SPAWNFLAG_NOTOUCH)
|
|
objerror ("health and notouch don't make sense\n");
|
|
self.max_health = self.health;
|
|
self.th_die = multi_killed;
|
|
self.takedamage = DAMAGE_YES;
|
|
self.solid = SOLID_BBOX;
|
|
}
|
|
else
|
|
{
|
|
if ( !(self.spawnflags & SPAWNFLAG_NOTOUCH) )
|
|
{
|
|
self.touch = multi_touch;
|
|
self.solid = SOLID_TRIGGER;
|
|
}
|
|
}
|
|
settriggermodel (self, self.model);
|
|
|
|
if (!self.target)
|
|
{
|
|
if (!self.message)
|
|
error ("There is no target set!");
|
|
}
|
|
|
|
};
|
|
|
|
|
|
/*QUAKED trigger_once (.5 .5 .5) ? notouch
|
|
Variable sized trigger. Triggers once, then removes itself. You must set the key "target" to the name of another object in the level that has a matching
|
|
"targetname". If "health" is set, the trigger must be killed to activate.
|
|
If notouch is set, the trigger is only fired by other entities, not by touching.
|
|
if "killtarget" is set, any objects that have a matching "target" will be removed when the trigger is fired.
|
|
if "angle" is set, the trigger will only fire when someone is facing the direction of the angle. Use "360" for an angle of 0.
|
|
sounds
|
|
1) secret
|
|
2) beep beep
|
|
3) large switch
|
|
4)
|
|
set "message" to text string
|
|
*/
|
|
void() trigger_once =
|
|
{
|
|
self.wait = -1;
|
|
trigger_multiple();
|
|
};
|
|
|
|
/*QUAKED trigger_secret (.5 .5 .5) ?
|
|
secret counter trigger
|
|
sounds
|
|
1) secret
|
|
2) beep beep
|
|
3)
|
|
4)
|
|
set "message" to text string
|
|
*/
|
|
void() trigger_secret =
|
|
{
|
|
total_secrets = total_secrets + 1;
|
|
self.wait = -1;
|
|
// self.classname = "trigger_secret";
|
|
if (!self.message)
|
|
self.message = "You found a secret area!";
|
|
if (!self.sounds)
|
|
self.sounds = 1;
|
|
|
|
if (self.sounds == 1)
|
|
{
|
|
precache_sound ("temp/secret.wav");
|
|
self.noise = "temp/secret.wav";
|
|
}
|
|
else if (self.sounds == 2)
|
|
{
|
|
precache_sound ("temp/talk.wav");
|
|
self.noise = "temp/talk.wav";
|
|
}
|
|
|
|
trigger_multiple ();
|
|
};
|
|
|
|
|
|
|
|
void() counter_use =
|
|
{
|
|
local string junk;
|
|
|
|
self.count = self.count - 1;
|
|
if (self.count < 0)
|
|
return;
|
|
|
|
if (self.count != 0)
|
|
{
|
|
if (activator.classname == "player"
|
|
&& (self.spawnflags & SPAWNFLAG_NOMESSAGE) == 0)
|
|
{
|
|
if (self.count > 4)
|
|
centerprint (activator, "There are mroe to go...");
|
|
else if (self.count == 3)
|
|
centerprint (activator, "Only 3 more to go...");
|
|
else if (self.count == 2)
|
|
centerprint (activator, "Only 2 more to go...");
|
|
else
|
|
centerprint (activator, "Only 1 more to go...");
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (activator.classname == "player"
|
|
&& (self.spawnflags & SPAWNFLAG_NOMESSAGE) == 0)
|
|
centerprint(activator, "Sequence completed!");
|
|
self.enemy = activator;
|
|
multi_trigger ();
|
|
};
|
|
|
|
/*QUAKED trigger_counter (.5 .5 .5) ? nomessage
|
|
Acts as an intermediary for an action that takes multiple inputs.
|
|
|
|
If nomessage is not set, t will print "1 more.. " etc when triggered and "sequence complete" when finished.
|
|
|
|
After the counter has been triggered "count" times (default 2), it will fire all of it's targets and remove itself.
|
|
*/
|
|
void() trigger_counter =
|
|
{
|
|
self.wait = -1;
|
|
if (!self.count)
|
|
self.count = 2;
|
|
|
|
if (!self.target)
|
|
error ("There is no target set!");
|
|
self.use = counter_use;
|
|
};
|
|
|
|
|
|
/*
|
|
==============================================================================
|
|
|
|
TELEPORT TRIGGERS
|
|
|
|
==============================================================================
|
|
*/
|
|
|
|
float PLAYER_ONLY = 1;
|
|
|
|
void() random_telesound =
|
|
{
|
|
local float v;
|
|
local string tmpstr;
|
|
|
|
v = random() * 5;
|
|
if (v < 1)
|
|
tmpstr = "misc/r_tele1.wav";
|
|
else if (v < 2)
|
|
tmpstr = "misc/r_tele2.wav";
|
|
else if (v < 3)
|
|
tmpstr = "misc/r_tele3.wav";
|
|
else if (v < 4)
|
|
tmpstr = "misc/r_tele4.wav";
|
|
else
|
|
tmpstr = "misc/r_tele5.wav";
|
|
|
|
sound (self, 1, tmpstr, 1, 1);
|
|
};
|
|
|
|
void() tfog1 = [ 0, tfog2 ] {};
|
|
void() tfog2 = [ 1, tfog3 ] {random_telesound();};
|
|
void() tfog3 = [ 2, tfog4 ] {};
|
|
void() tfog4 = [ 3, tfog5 ] {};
|
|
void() tfog5 = [ 4, tfog6 ] {};
|
|
void() tfog6 = [ 5, tfog7 ] {};
|
|
void() tfog7 = [ 6, tfog8 ] {};
|
|
void() tfog8 = [ 7, tfog9 ] {};
|
|
void() tfog9 = [ 8, tfog10 ] {};
|
|
void() tfog10 = [ 9, tfog11 ] {};
|
|
void() tfog11 = [ 9, tfog11 ] {remove(self);};
|
|
|
|
void(vector org) spawn_tfog =
|
|
{
|
|
s = spawn ();
|
|
s.origin = org;
|
|
s.angles = '0 0 0';
|
|
s.movetype = MOVETYPE_NONE;
|
|
s.solid = SOLID_NOT;
|
|
setmodel (s, "sprites/s_telep.spr");
|
|
|
|
old = self;
|
|
self = s;
|
|
tfog1 ();
|
|
self = old;
|
|
};
|
|
|
|
|
|
void() tdeath_touch =
|
|
{
|
|
if (other == self.owner)
|
|
return;
|
|
|
|
if (other.health)
|
|
{
|
|
self.solid = SOLID_NOT;
|
|
T_Damage (other, self, self, 1000);
|
|
}
|
|
};
|
|
|
|
void() tdeath_remove =
|
|
{
|
|
remove (self);
|
|
return;
|
|
};
|
|
|
|
void(vector org, entity death_owner) spawn_tdeath =
|
|
{
|
|
local entity death;
|
|
|
|
death = spawn();
|
|
death.classname = "teledeath";
|
|
death.origin = org;
|
|
death.movetype = MOVETYPE_NONE;
|
|
death.solid = SOLID_TRIGGER;
|
|
death.angles = '0 0 0';
|
|
// FIX ME (this does not set the size properly)
|
|
setsize (death, '-20 -20 -20', '20 20 20');
|
|
death.touch = tdeath_touch;
|
|
death.nextthink = time + 0.1;
|
|
death.think = tdeath_remove;
|
|
death.owner = death_owner;
|
|
};
|
|
|
|
void() teleport_touch =
|
|
{
|
|
local entity t;
|
|
local vector org;
|
|
|
|
|
|
if (self.spawnflags & PLAYER_ONLY)
|
|
{
|
|
if (other.classname != "player")
|
|
return;
|
|
}
|
|
|
|
// only teleport living creatures
|
|
if (other.health <= 0)
|
|
return;
|
|
|
|
// put a tfog where the player was
|
|
spawn_tfog (other.origin);
|
|
|
|
// FIXME: precalc at awake time
|
|
t = find (world, targetname, self.target);
|
|
if (!t)
|
|
objerror ("couldn't find target");
|
|
|
|
// spawn a tfog flash in front of the destination
|
|
makevectors (t.mangle);
|
|
org = t.origin + 32 * v_forward;
|
|
|
|
spawn_tfog (org);
|
|
spawn_tdeath(t.origin, other);
|
|
|
|
// move the player and lock him down for a little while
|
|
if (!other.health)
|
|
{
|
|
other.origin = t.origin;
|
|
other.velocity = (v_forward * other.velocity_x) + (v_forward * other.velocity_y);
|
|
return;
|
|
}
|
|
|
|
setorigin (other, t.origin);
|
|
other.angles = t.mangle;
|
|
if (other.classname == "player")
|
|
{
|
|
other.fixangle = 1; // turn this way immediately
|
|
other.teleport_time = time + 0.7;
|
|
if (other.flags & FL_ONGROUND)
|
|
other.flags = other.flags - FL_ONGROUND;
|
|
other.velocity = v_forward * 300;
|
|
}
|
|
other.flags = other.flags - other.flags & FL_ONGROUND;
|
|
};
|
|
|
|
/*QUAKED info_teleport_destination (.5 .5 .5) (-8 -8 -8) (8 8 32)
|
|
This is the destination marker for a teleporter. It should have a "targetname" field with the same value as a teleporter's "target" field.
|
|
*/
|
|
void() info_teleport_destination =
|
|
{
|
|
// this does nothing, just serves as a target spot
|
|
self.mangle = self.angles;
|
|
self.angles = '0 0 0';
|
|
self.model = "";
|
|
self.origin = self.origin + '0 0 27';
|
|
if (!self.targetname)
|
|
objerror ("no targetname");
|
|
};
|
|
|
|
|
|
/*QUAKED trigger_teleport (.5 .5 .5) ? PLAYER_ONLY
|
|
Any object touching this will be transported to the corresponding info_teleport_destination entity. You must set the "target" field, and create an object with a "targetname" field that matches.
|
|
*/
|
|
void() trigger_teleport =
|
|
{
|
|
self.mangle = self.angles;
|
|
self.angles = '0 0 0';
|
|
setsize (self, self.mins, self.maxs);
|
|
|
|
self.solid = SOLID_TRIGGER;
|
|
settriggermodel (self, self.model);
|
|
self.touch = teleport_touch;
|
|
self.angles = '0 0 0';
|
|
// find the destination
|
|
if (!self.target)
|
|
objerror ("no target");
|
|
};
|
|
|
|
/*
|
|
==============================================================================
|
|
|
|
trigger_setskill
|
|
|
|
==============================================================================
|
|
*/
|
|
|
|
void() trigger_skill_touch =
|
|
{
|
|
if (other.classname != "player")
|
|
return;
|
|
|
|
cvar_set ("skill", self.message);
|
|
};
|
|
|
|
/*QUAKED trigger_setskill (.5 .5 .5) ?
|
|
sets skill level to the value of "message".
|
|
Only used on start map.
|
|
*/
|
|
void() trigger_setskill =
|
|
{
|
|
self.mangle = self.angles;
|
|
self.angles = '0 0 0';
|
|
setsize (self, self.mins, self.maxs);
|
|
|
|
self.solid = SOLID_TRIGGER;
|
|
settriggermodel (self, self.model);
|
|
self.touch = trigger_skill_touch;
|
|
self.angles = '0 0 0';
|
|
};
|
|
|
|
|
|
/*
|
|
==============================================================================
|
|
|
|
ONLY REGISTERED TRIGGERS
|
|
|
|
==============================================================================
|
|
*/
|
|
|
|
void() trigger_onlyregistered_touch =
|
|
{
|
|
if (self.attack_finished > time)
|
|
return;
|
|
|
|
self.attack_finished = time + 2;
|
|
if (cvar("registered"))
|
|
{
|
|
SUB_UseTargets ();
|
|
remove (self);
|
|
}
|
|
else
|
|
{
|
|
centerprint (other, self.message);
|
|
sound (other, 0, "temp/talk.wav", 1,1);
|
|
}
|
|
};
|
|
|
|
/*QUAKED trigger_onlyregistered (.5 .5 .5) ?
|
|
Only fires if playing the registered version, otherwise prints the message
|
|
*/
|
|
void() trigger_onlyregistered =
|
|
{
|
|
self.mangle = self.angles;
|
|
self.angles = '0 0 0';
|
|
setsize (self, self.mins, self.maxs);
|
|
|
|
precache_sound ("temp/talk.wav");
|
|
|
|
self.solid = SOLID_TRIGGER;
|
|
settriggermodel (self, self.model);
|
|
self.touch = trigger_onlyregistered_touch;
|
|
self.angles = '0 0 0';
|
|
};
|
|
|
|
//============================================================================
|
|
|
|
void() hurt_on =
|
|
{
|
|
self.solid = SOLID_TRIGGER;
|
|
self.nextthink = -1;
|
|
};
|
|
|
|
void() hurt_touch =
|
|
{
|
|
if (other.health)
|
|
{
|
|
self.solid = SOLID_NOT;
|
|
T_Damage (other, self, self, self.dmg);
|
|
self.think = hurt_on;
|
|
self.nextthink = time + 1;
|
|
}
|
|
|
|
return;
|
|
};
|
|
|
|
/*QUAKED trigger_hurt (.5 .5 .5) ?
|
|
Any object touching this will be hurt
|
|
set dmg to damage amount
|
|
defalt dmg = 5
|
|
*/
|
|
void() trigger_hurt =
|
|
{
|
|
self.mangle = self.angles;
|
|
self.angles = '0 0 0';
|
|
|
|
self.solid = SOLID_TRIGGER;
|
|
settriggermodel (self, self.model);
|
|
self.touch = hurt_touch;
|
|
self.angles = '0 0 0';
|
|
if (!self.dmg)
|
|
self.dmg = 5;
|
|
};
|
|
|
|
|
|
/*QUAKED trigger_push (.5 .5 .5) ? PUSH_ONCE
|
|
Pushes the player
|
|
*/
|
|
float PUSH_ONCE = 1;
|
|
void() trigger_push_touch;
|
|
void() trigger_push =
|
|
{
|
|
|
|
self.movetype = MOVETYPE_NONE;
|
|
self.solid = SOLID_TRIGGER;
|
|
settriggermodel (self, self.model);
|
|
self.touch = trigger_push_touch;
|
|
SetMovedir();
|
|
};
|
|
|
|
void() trigger_push_touch =
|
|
{
|
|
if (other.classname == "player")
|
|
other.velocity = 1000 * self.movedir + 400 * '0 0 1';
|
|
if (self.spawnflags & PUSH_ONCE)
|
|
remove(self);
|
|
};
|
|
|
|
//============================================================================
|
|
|
|
void() trigger_monsterjump_touch =
|
|
{
|
|
if ( other.flags & (FL_MONSTER | FL_FLY | FL_SWIM) != FL_MONSTER )
|
|
return;
|
|
|
|
// set XY even if not on ground, so the jump will clear lips
|
|
other.velocity_x = self.movedir_x * self.speed;
|
|
other.velocity_y = self.movedir_y * self.speed;
|
|
|
|
if ( !(other.flags & FL_ONGROUND) )
|
|
return;
|
|
|
|
other.flags = other.flags - FL_ONGROUND;
|
|
|
|
other.velocity_z = self.height;
|
|
};
|
|
|
|
/*QUAKED trigger_monsterjump (.5 .5 .5) ?
|
|
Walking monsters that touch this will jump in the direction of the trigger's angle
|
|
"speed" default to 200, the speed thrown forward
|
|
"height" default to 200, the speed thrown upwards
|
|
*/
|
|
void() trigger_monsterjump =
|
|
{
|
|
if (!self.speed)
|
|
self.speed = 200;
|
|
if (!self.height)
|
|
self.height = 200;
|
|
self.movetype = MOVETYPE_NONE;
|
|
self.solid = SOLID_TRIGGER;
|
|
settriggermodel (self, self.model);
|
|
self.touch = trigger_monsterjump_touch;
|
|
SetMovedir();
|
|
};
|
|
|