mirror of
https://github.com/DrBeef/JKXR.git
synced 2025-01-25 09:41:34 +00:00
269 lines
7.9 KiB
C++
269 lines
7.9 KiB
C++
|
/*
|
||
|
===========================================================================
|
||
|
Copyright (C) 2000 - 2013, Raven Software, Inc.
|
||
|
Copyright (C) 2001 - 2013, Activision, Inc.
|
||
|
Copyright (C) 2013 - 2015, OpenJK contributors
|
||
|
|
||
|
This file is part of the OpenJK source code.
|
||
|
|
||
|
OpenJK is free software; you can redistribute it and/or modify it
|
||
|
under the terms of the GNU General Public License version 2 as
|
||
|
published by the Free Software Foundation.
|
||
|
|
||
|
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, see <http://www.gnu.org/licenses/>.
|
||
|
===========================================================================
|
||
|
*/
|
||
|
|
||
|
#include "g_local.h"
|
||
|
#include "g_functions.h"
|
||
|
|
||
|
extern void InitMover( gentity_t *ent );
|
||
|
|
||
|
extern gentity_t *G_TestEntityPosition( gentity_t *ent );
|
||
|
void func_wait_return_solid( gentity_t *self )
|
||
|
{
|
||
|
//once a frame, see if it's clear.
|
||
|
self->clipmask = CONTENTS_BODY;//|CONTENTS_MONSTERCLIP|CONTENTS_BOTCLIP;
|
||
|
if ( !(self->spawnflags&16) || G_TestEntityPosition( self ) == NULL )
|
||
|
{
|
||
|
gi.SetBrushModel( self, self->model );
|
||
|
VectorCopy( self->currentOrigin, self->pos1 );
|
||
|
InitMover( self );
|
||
|
/*
|
||
|
VectorCopy( self->s.origin, self->s.pos.trBase );
|
||
|
VectorCopy( self->s.origin, self->currentOrigin );
|
||
|
*/
|
||
|
//if we moved, we want the *current* origin, not our start origin!
|
||
|
VectorCopy( self->currentOrigin, self->s.pos.trBase );
|
||
|
gi.linkentity( self );
|
||
|
self->svFlags &= ~SVF_NOCLIENT;
|
||
|
self->s.eFlags &= ~EF_NODRAW;
|
||
|
self->e_UseFunc = useF_func_usable_use;
|
||
|
self->clipmask = 0;
|
||
|
if ( self->target2 && self->target2[0] )
|
||
|
{
|
||
|
G_UseTargets2( self, self->activator, self->target2 );
|
||
|
}
|
||
|
if ( self->s.eFlags & EF_ANIM_ONCE )
|
||
|
{//Start our anim
|
||
|
self->s.frame = 0;
|
||
|
}
|
||
|
//NOTE: be sure to reset the brushmodel before doing this or else CONTENTS_OPAQUE may not be on when you call this
|
||
|
if ( !(self->spawnflags&1) )
|
||
|
{//START_OFF doesn't effect area portals
|
||
|
gi.AdjustAreaPortalState( self, qfalse );
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
self->clipmask = 0;
|
||
|
self->e_ThinkFunc = thinkF_func_wait_return_solid;
|
||
|
self->nextthink = level.time + FRAMETIME;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void func_usable_think( gentity_t *self )
|
||
|
{
|
||
|
if ( self->spawnflags & 8 )
|
||
|
{
|
||
|
self->svFlags |= SVF_PLAYER_USABLE; //Replace the usable flag
|
||
|
self->e_UseFunc = useF_func_usable_use;
|
||
|
self->e_ThinkFunc = thinkF_NULL;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
qboolean G_EntIsRemovableUsable( int entNum )
|
||
|
{
|
||
|
gentity_t *ent = &g_entities[entNum];
|
||
|
if ( ent->classname && !Q_stricmp( "func_usable", ent->classname ) )
|
||
|
{
|
||
|
if ( !(ent->s.eFlags&EF_SHADER_ANIM) && !(ent->spawnflags&8) && ent->targetname )
|
||
|
{//not just a shader-animator and not ALWAYS_ON, so it must be removable somehow
|
||
|
return qtrue;
|
||
|
}
|
||
|
}
|
||
|
return qfalse;
|
||
|
}
|
||
|
|
||
|
void func_usable_use( gentity_t *self, gentity_t *other, gentity_t *activator )
|
||
|
{//Toggle on and off
|
||
|
if ( other == activator )
|
||
|
{//directly used by use button trace
|
||
|
if ( (self->spawnflags&32) )
|
||
|
{//only usable by NPCs
|
||
|
if ( activator->NPC == NULL )
|
||
|
{//Not an NPC
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
G_ActivateBehavior( self, BSET_USE );
|
||
|
if ( self->s.eFlags & EF_SHADER_ANIM )
|
||
|
{//animate shader when used
|
||
|
self->s.frame++;//inc frame
|
||
|
if ( self->s.frame > self->endFrame )
|
||
|
{//wrap around
|
||
|
self->s.frame = 0;
|
||
|
}
|
||
|
if ( self->target && self->target[0] )
|
||
|
{
|
||
|
G_UseTargets( self, activator );
|
||
|
}
|
||
|
}
|
||
|
else if ( self->spawnflags & 8 )
|
||
|
{//ALWAYS_ON
|
||
|
//Remove the ability to use the entity directly
|
||
|
self->svFlags &= ~SVF_PLAYER_USABLE;
|
||
|
//also remove ability to call any use func at all!
|
||
|
self->e_UseFunc = useF_NULL;
|
||
|
|
||
|
if(self->target && self->target[0])
|
||
|
{
|
||
|
G_UseTargets(self, activator);
|
||
|
}
|
||
|
|
||
|
if ( self->wait )
|
||
|
{
|
||
|
self->e_ThinkFunc = thinkF_func_usable_think;
|
||
|
self->nextthink = level.time + ( self->wait * 1000 );
|
||
|
}
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
else if ( !self->count )
|
||
|
{//become solid again
|
||
|
self->count = 1;
|
||
|
self->activator = activator;
|
||
|
func_wait_return_solid( self );
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
//NOTE: MUST do this BEFORE clearing contents, or you may not open the area portal!!!
|
||
|
if ( !(self->spawnflags&1) )
|
||
|
{//START_OFF doesn't effect area portals
|
||
|
gi.AdjustAreaPortalState( self, qtrue );
|
||
|
}
|
||
|
self->s.solid = 0;
|
||
|
self->contents = 0;
|
||
|
self->clipmask = 0;
|
||
|
self->svFlags |= SVF_NOCLIENT;
|
||
|
self->s.eFlags |= EF_NODRAW;
|
||
|
self->count = 0;
|
||
|
|
||
|
if(self->target && self->target[0])
|
||
|
{
|
||
|
G_UseTargets(self, activator);
|
||
|
}
|
||
|
self->e_ThinkFunc = thinkF_NULL;
|
||
|
self->nextthink = -1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void func_usable_pain(gentity_t *self, gentity_t *inflictor, gentity_t *attacker, const vec3_t point, int damage, int mod,int hitLoc)
|
||
|
{
|
||
|
if ( self->paintarget )
|
||
|
{
|
||
|
G_UseTargets2 (self, self->activator, self->paintarget);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
GEntity_UseFunc( self, attacker, attacker );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void func_usable_die(gentity_t *self, gentity_t *inflictor, gentity_t *attacker, int damage, int mod,int dFlags,int hitLoc)
|
||
|
{
|
||
|
self->takedamage = qfalse;
|
||
|
GEntity_UseFunc( self, inflictor, attacker );
|
||
|
}
|
||
|
|
||
|
/*QUAKED func_usable (0 .5 .8) ? STARTOFF AUTOANIMATE ANIM_ONCE ALWAYS_ON BLOCKCHECK NPC_USE PLAYER_USE INACTIVE
|
||
|
START_OFF - the wall will not be there
|
||
|
AUTOANIMATE - if a model is used it will animate
|
||
|
ANIM_ONCE - When turned on, goes through anim once
|
||
|
ALWAYS_ON - Doesn't toggle on and off when used, just runs usescript and fires target
|
||
|
NPC_ONLY - Only NPCs can directly use this
|
||
|
PLAYER_USE - Player can use it with the use button
|
||
|
BLOCKCHECK - Will not turn on while something is inside it
|
||
|
|
||
|
A bmodel that just sits there, doing nothing. Can be used for conditional walls and models.
|
||
|
"targetname" - When used, will toggle on and off
|
||
|
"target" Will fire this target every time it is toggled OFF
|
||
|
"target2" Will fire this target every time it is toggled ON
|
||
|
"model2" .md3 model to also draw
|
||
|
"modelAngles" md3 model's angles <pitch yaw roll> (in addition to any rotation on the part of the brush entity itself)
|
||
|
"color" constantLight color
|
||
|
"light" constantLight radius
|
||
|
"usescript" script to run when turned on
|
||
|
"deathscript" script to run when turned off
|
||
|
"wait" amount of time before the object is usable again (only valid with ALWAYS_ON flag)
|
||
|
"health" if it has health, it will be used whenever shot at/killed - if you want it to only be used once this way, set health to 1
|
||
|
"endframe" Will make it animate to next shader frame when used, not turn on/off... set this to number of frames in the shader, minus 1
|
||
|
"forcevisible" - When you turn on force sight (any level), you can see these draw through the entire level...
|
||
|
*/
|
||
|
|
||
|
void SP_func_usable( gentity_t *self )
|
||
|
{
|
||
|
gi.SetBrushModel( self, self->model );
|
||
|
InitMover( self );
|
||
|
VectorCopy( self->s.origin, self->s.pos.trBase );
|
||
|
VectorCopy( self->s.origin, self->currentOrigin );
|
||
|
VectorCopy( self->s.origin, self->pos1 );
|
||
|
|
||
|
self->count = 1;
|
||
|
if (self->spawnflags & 1)
|
||
|
{
|
||
|
self->spawnContents = self->contents; // It Navs can temporarly turn it "on"
|
||
|
self->s.solid = 0;
|
||
|
self->contents = 0;
|
||
|
self->clipmask = 0;
|
||
|
self->svFlags |= SVF_NOCLIENT;
|
||
|
self->s.eFlags |= EF_NODRAW;
|
||
|
self->count = 0;
|
||
|
}
|
||
|
|
||
|
if (self->spawnflags & 2)
|
||
|
{
|
||
|
self->s.eFlags |= EF_ANIM_ALLFAST;
|
||
|
}
|
||
|
|
||
|
if (self->spawnflags & 4)
|
||
|
{//FIXME: need to be able to do change to something when it's done? Or not be usable until it's done?
|
||
|
self->s.eFlags |= EF_ANIM_ONCE;
|
||
|
}
|
||
|
|
||
|
self->e_UseFunc = useF_func_usable_use;
|
||
|
|
||
|
if ( self->health )
|
||
|
{
|
||
|
self->takedamage = qtrue;
|
||
|
self->e_DieFunc = dieF_func_usable_die;
|
||
|
self->e_PainFunc = painF_func_usable_pain;
|
||
|
}
|
||
|
|
||
|
if ( self->endFrame > 0 )
|
||
|
{
|
||
|
self->s.frame = self->startFrame = 0;
|
||
|
self->s.eFlags |= EF_SHADER_ANIM;
|
||
|
}
|
||
|
|
||
|
gi.linkentity (self);
|
||
|
|
||
|
int forceVisible = 0;
|
||
|
G_SpawnInt( "forcevisible", "0", &forceVisible );
|
||
|
if ( forceVisible )
|
||
|
{//can see these through walls with force sight, so must be broadcast
|
||
|
if ( VectorCompare( self->s.origin, vec3_origin ) )
|
||
|
{//no origin brush
|
||
|
self->svFlags |= SVF_BROADCAST;
|
||
|
}
|
||
|
self->s.eFlags |= EF_FORCE_VISIBLE;
|
||
|
}
|
||
|
}
|