jkxr/Projects/Android/jni/OpenJK/codeJK2/game/g_usable.cpp
Simon 4597b03873 Initial Commit
Opens in Android Studio but haven't even tried to build it yet (it won't.. I know that much!)
2022-09-18 16:37:21 +01:00

249 lines
No EOL
7 KiB
C++

/*
===========================================================================
Copyright (C) 1999 - 2005, Id Software, Inc.
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_headers.h"
#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;
}
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
{
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;
if ( !(self->spawnflags&1) )
{//START_OFF doesn't effect area portals
gi.AdjustAreaPortalState( self, qtrue );
}
}
}
void func_usable_pain(gentity_t *self, gentity_t *inflictor, gentity_t *attacker, vec3_t point, int damage, int mod,int hitLoc)
{
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
"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
*/
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->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);
}