mirror of
https://github.com/nzp-team/quakec.git
synced 2025-01-07 10:21:09 +00:00
437 lines
9.8 KiB
C++
437 lines
9.8 KiB
C++
/*
|
|
server/entities/triggers.qc
|
|
|
|
Misc. Trigger functions
|
|
|
|
Copyright (C) 2021-2024 NZ:P Team
|
|
|
|
This program is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU General Public License
|
|
as published by the Free Software Foundation; either version 2
|
|
of the License, or (at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
See the GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to:
|
|
|
|
Free Software Foundation, Inc.
|
|
59 Temple Place - Suite 330
|
|
Boston, MA 02111-1307, USA
|
|
|
|
*/
|
|
|
|
#define SPAWNFLAG_NOMESSAGE 1
|
|
#define SPAWNFLAG_NOTOUCH 1
|
|
|
|
void() Zombie_ReassignSpawnIDs;
|
|
|
|
// 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;
|
|
}
|
|
};
|
|
|
|
|
|
// 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
|
|
}
|
|
|
|
// don't trigger again until reset
|
|
self.takedamage = DAMAGE_NO;
|
|
|
|
activator = self.enemy;
|
|
|
|
SUB_UseTargets();
|
|
|
|
if (self.wait > 0)
|
|
{
|
|
self.think = multi_wait;
|
|
self.nextthink = time + self.wait;
|
|
}
|
|
else
|
|
{
|
|
Ent_FakeRemove(self);
|
|
}
|
|
};
|
|
|
|
void() multi_killed =
|
|
{
|
|
// motolegacy - FIXME
|
|
//self.enemy = damage_attacker;
|
|
multi_trigger();
|
|
};
|
|
|
|
void() multi_use =
|
|
{
|
|
self.enemy = activator;
|
|
multi_trigger();
|
|
};
|
|
|
|
void() multi_touch =
|
|
{
|
|
if (other.classname != "player")
|
|
return;
|
|
|
|
// if the trigger has an angles field, check player's facing direction
|
|
if (self.movedir != '0 0 0')
|
|
{
|
|
makevectors (other.angles);
|
|
if (v_forward * self.movedir < 0)
|
|
return; // not facing the right way
|
|
}
|
|
|
|
self.enemy = other;
|
|
multi_trigger();
|
|
};
|
|
|
|
void() SetMovedir =
|
|
{
|
|
if (self.angles == '0 -1 0')
|
|
self.movedir = '0 0 1';
|
|
else if (self.angles == '0 -2 0')
|
|
self.movedir = '0 0 -1';
|
|
else
|
|
{
|
|
makevectors (self.angles);
|
|
self.movedir = v_forward;
|
|
}
|
|
|
|
self.angles = '0 0 0';
|
|
};
|
|
|
|
void() InitTrigger =
|
|
{
|
|
// trigger angles are used for one-way touches. An angle of 0 is assumed
|
|
// to mean no restrictions, so use a yaw of 360 instead.
|
|
if (self.angles != '0 0 0')
|
|
SetMovedir ();
|
|
self.solid = SOLID_TRIGGER;
|
|
setmodel (self, self.model); // set size and link into world
|
|
self.movetype = MOVETYPE_NONE;
|
|
self.modelindex = 0;
|
|
self.model = "";
|
|
};
|
|
|
|
entity last_act_trigger;
|
|
void() trigger_activator_touch =
|
|
{
|
|
other.cost = other.cost +1; //hack, we can only touch one of thease at the time
|
|
if (other.classname != "player" || last_act_trigger == self || other.cost > 1)
|
|
return;
|
|
last_act_trigger = self;
|
|
|
|
entity t;
|
|
float tempcount, temptotal,breakthis;
|
|
string tempstring;
|
|
temptotal = 0;
|
|
breakthis = 0;
|
|
tempcount = 1;
|
|
tempstring = "";
|
|
t = world;
|
|
|
|
if (self.target2)
|
|
tempcount =+ 1;
|
|
if (self.target3)
|
|
tempcount =+ 1;
|
|
if (self.target4)
|
|
tempcount =+ 1;
|
|
if (self.target5)
|
|
tempcount =+ 1;
|
|
if (self.target6)
|
|
tempcount =+ 1;
|
|
if (self.target7)
|
|
tempcount =+ 1;
|
|
if (self.target8)
|
|
tempcount =+ 1;
|
|
if (self.target2)
|
|
tempcount = tempcount + 1;
|
|
if (self.target3)
|
|
tempcount = tempcount + 1;
|
|
if (self.target4)
|
|
tempcount = tempcount + 1;
|
|
if (self.target5)
|
|
tempcount = tempcount + 1;
|
|
if (self.target6)
|
|
tempcount = tempcount + 1;
|
|
if (self.target7)
|
|
tempcount = tempcount + 1;
|
|
if (self.target8)
|
|
tempcount = tempcount + 1;
|
|
|
|
while(tempcount > temptotal)
|
|
{
|
|
temptotal = temptotal + 1;
|
|
if (temptotal == 1)
|
|
tempstring = self.target;
|
|
if (temptotal == 2)
|
|
tempstring = self.target2;
|
|
if (temptotal == 3)
|
|
tempstring = self.target3;
|
|
if (temptotal == 4)
|
|
tempstring = self.target4;
|
|
if (temptotal == 5)
|
|
tempstring = self.target5;
|
|
if (temptotal == 6)
|
|
tempstring = self.target6;
|
|
if (temptotal == 7)
|
|
tempstring = self.target7;
|
|
if (temptotal == 8)
|
|
tempstring = self.target8;
|
|
if (tempstring)
|
|
{
|
|
t = find (world, targetname, tempstring);
|
|
breakthis = 0;
|
|
while (!breakthis)
|
|
{
|
|
if (!t)
|
|
{
|
|
breakthis = true;
|
|
}
|
|
if (t.classname == "spawn_zombie_away")
|
|
{
|
|
t.classname = "spawn_zombie";
|
|
t.spawn_id = zombie_spawn_points;
|
|
zombie_spawn_points++;
|
|
/*if (cvar("developer"))
|
|
setmodel(t, "progs/ai/zfull.mdl");*/
|
|
}
|
|
t = find (t, targetname, tempstring);
|
|
}
|
|
}
|
|
}
|
|
Zombie_ReassignSpawnIDs();
|
|
}
|
|
|
|
void() trigger_activator =
|
|
{
|
|
InitTrigger ();
|
|
self.touch = trigger_activator_touch;
|
|
}
|
|
|
|
void() trigger_interact_touch =
|
|
{
|
|
if (other.classname != "player" || other.downed || other.isBuying == true || !PlayerIsLooking(other, self))
|
|
return;
|
|
|
|
if (other.button7) {
|
|
if (self.noise)
|
|
Sound_PlaySound(other, self.noise, SOUND_TYPE_ENV_OBJECT, SOUND_PRIORITY_PLAYALWAYS);
|
|
multi_trigger();
|
|
}
|
|
}
|
|
|
|
void() trigger_interact =
|
|
{
|
|
InitTrigger();
|
|
self.touch = trigger_interact_touch;
|
|
}
|
|
|
|
/* ===================
|
|
Custom Teddy Triggers
|
|
===================*/
|
|
|
|
void() teddy_react =
|
|
{
|
|
local entity t;
|
|
if (self.spawnflags & 1) {
|
|
t = find (world, teddyremovetarget, self.target);
|
|
|
|
if (t)
|
|
Ent_FakeRemove(t);
|
|
} else if (self.target) {
|
|
SUB_UseTargets();
|
|
}
|
|
|
|
entity dummy = spawn();
|
|
setorigin(dummy, self.origin);
|
|
dummy.think = SUB_Remove;
|
|
dummy.nextthink = time + 10;
|
|
|
|
|
|
if (self.noise)
|
|
Sound_PlaySound(dummy, self.noise, SOUND_TYPE_ENV_OBJECT, SOUND_PRIORITY_PLAYALWAYS);
|
|
|
|
Ent_FakeRemove(self);
|
|
}
|
|
|
|
/*
|
|
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.
|
|
NOTOUCH has been obsoleted by trigger_relay!
|
|
set "message" to text string
|
|
*/
|
|
void() trigger_multiple =
|
|
{
|
|
if (!self.wait)
|
|
self.wait = 0.2;
|
|
|
|
self.use = multi_use;
|
|
|
|
InitTrigger ();
|
|
|
|
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;
|
|
setorigin(self, self.origin); // make sure it links into the world
|
|
}
|
|
else
|
|
{
|
|
if (!(self.spawnflags & SPAWNFLAG_NOTOUCH))
|
|
{
|
|
self.touch = multi_touch;
|
|
}
|
|
}
|
|
};
|
|
|
|
/*
|
|
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.
|
|
set "message" to text string
|
|
*/
|
|
void() trigger_once =
|
|
{
|
|
self.wait = -1;
|
|
|
|
trigger_multiple();
|
|
}
|
|
|
|
/*QUAKED trigger_relay (.5 .5 .5) (-8 -8 -8) (8 8 8)
|
|
This fixed size trigger cannot be touched, it can only be fired by other events. It can contain killtargets, targets, delays, and messages.
|
|
*/
|
|
void() trigger_relay =
|
|
{
|
|
self.use = SUB_UseTargets;
|
|
};
|
|
|
|
void() trigger_atroundend =
|
|
{
|
|
self.classname = "trigger_atroundend";
|
|
//self.use = multi_use;
|
|
}
|
|
|
|
#define SPAWNFLAG_TRIGGERFIRE_PLAYER 1
|
|
#define SPAWNFLAG_TRIGGERFIRE_NODAMAGE 2
|
|
#define SPAWNFLAG_TRIGGERFIRE_HEALTHNERF 4
|
|
|
|
void() trigger_setfire_touch =
|
|
{
|
|
if (other.aistatus != "1" && (other.classname == "player" && !(self.spawnflags & SPAWNFLAG_TRIGGERFIRE_PLAYER)))
|
|
return;
|
|
|
|
other.onfire = true;
|
|
other.fire_timeout = time + self.fire_timeout;
|
|
|
|
if ((self.spawnflags & SPAWNFLAG_TRIGGERFIRE_NODAMAGE) && other.aistatus == "1")
|
|
other.ltime = time + 10000;
|
|
}
|
|
|
|
void() trigger_setfire =
|
|
{
|
|
InitTrigger ();
|
|
self.touch = trigger_setfire_touch;
|
|
}
|
|
|
|
//
|
|
// trigger_awardpoints
|
|
// Awards touching client Score on contact.
|
|
//
|
|
#define SPAWNFLAG_TRIGGERSCORE_REQUIRESTAND 1
|
|
#define SPAWNFLAG_TRIGGERSCORE_REQUIRECROUCH 2
|
|
#define SPAWNFLAG_TRIGGERSCORE_REQUIREPRONE 4
|
|
#define SPAWNFLAG_TRIGGERSCORE_APPLY2XPOINTS 8
|
|
|
|
void() trigger_awardpoints_touch =
|
|
{
|
|
if (other.classname != "player" || other.downed || !self.health)
|
|
return;
|
|
|
|
if (other.stance != PLAYER_STANCE_STAND && (self.spawnflags & SPAWNFLAG_TRIGGERSCORE_REQUIRESTAND))
|
|
return;
|
|
if (other.stance != PLAYER_STANCE_CROUCH && (self.spawnflags & SPAWNFLAG_TRIGGERSCORE_REQUIRECROUCH))
|
|
return;
|
|
if (other.stance != PLAYER_STANCE_PRONE && (self.spawnflags & SPAWNFLAG_TRIGGERSCORE_REQUIREPRONE))
|
|
return;
|
|
|
|
Player_AddScore(other, self.points, (self.spawnflags & SPAWNFLAG_TRIGGERSCORE_APPLY2XPOINTS));
|
|
|
|
if (self.noise != "")
|
|
Sound_PlaySound(self, self.noise, SOUND_TYPE_ENV_CHING, SOUND_PRIORITY_PLAYALWAYS);
|
|
|
|
self.health = 0;
|
|
}
|
|
|
|
void() trigger_awardpoints =
|
|
{
|
|
InitTrigger ();
|
|
self.touch = trigger_awardpoints_touch;
|
|
self.health = 1;
|
|
|
|
if (self.noise != "")
|
|
precache_sound(self.noise);
|
|
}
|
|
|
|
//
|
|
// Quake Triggers
|
|
//
|
|
|
|
// NZ:P START -- Spawnflags for not hurting clients and not hurting AI
|
|
#define HURT_SPAWNFLAG_NOAI 1
|
|
#define HURT_SPAWNFLAG_NOCLIENT 2
|
|
// NZ:P END
|
|
|
|
void() hurt_on =
|
|
{
|
|
self.solid = SOLID_TRIGGER;
|
|
self.nextthink = -1;
|
|
};
|
|
|
|
void() hurt_touch =
|
|
{
|
|
if ((other.classname == "player" && !(self.spawnflags & HURT_SPAWNFLAG_NOCLIENT)) ||
|
|
(other.aistatus == "1" && !(self.spawnflags & HURT_SPAWNFLAG_NOAI)))
|
|
{
|
|
self.solid = SOLID_NOT;
|
|
DamageHandler(other, self, self.dmg, DMG_TYPE_OTHER);
|
|
self.think = hurt_on;
|
|
self.nextthink = time + 1;
|
|
|
|
if (self.message)
|
|
centerprint(other, strcat(self.message, "\n"));
|
|
}
|
|
|
|
return;
|
|
};
|
|
|
|
void() trigger_hurt =
|
|
{
|
|
InitTrigger ();
|
|
self.touch = hurt_touch;
|
|
if (!self.dmg)
|
|
self.dmg = 5;
|
|
};
|