/* Copyright (C) 1997-2001 Id Software, Inc. 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 the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ // g_misc.c #include "g_local.h" //===================================================== //================= //Misc functions //================= void VelocityForDamage( int damage, vec3_t v ) { v[0] = 10.0 * crandom(); v[1] = 10.0 * crandom(); v[2] = 20.0 + 10.0 * random(); VectorNormalizeFast(v); if( damage < 50 ) VectorScale(v, 0.7, v); else VectorScale(v, 1.2, v); } void ClipGibVelocity (edict_t *ent) { if (ent->velocity[0] < -100) ent->velocity[0] = -100; else if (ent->velocity[0] > 100) ent->velocity[0] = 100; if (ent->velocity[1] < -100) ent->velocity[1] = -100; else if (ent->velocity[1] > 100) ent->velocity[1] = 100; if (ent->velocity[2] < 50) ent->velocity[2] = 50; // always some upwards else if (ent->velocity[2] > 150) ent->velocity[2] = 150; } //================= //gibs //================= void ThrowSmallPileOfGibs ( edict_t *self, int count, int damage ) { vec3_t origin, velocity; edict_t *event; int contents; contents = trap_PointContents( self->s.origin ); if (contents & CONTENTS_NODROP) return; VectorCopy(self->s.origin, origin); origin[2] -= 16; event = G_SpawnEvent( EV_SPOG, count, origin ); //transform damage into origin2 to scale speed at cgame VelocityForDamage (damage, velocity); VectorMA (self->velocity, 0.1, velocity, event->s.origin2); //clip gib velocity if (event->s.origin2[0] < -200) event->s.origin2[0] = -200; else if (event->s.origin2[0] > 200) event->s.origin2[0] = 200; if (event->s.origin2[1] < -200) event->s.origin2[1] = -200; else if (event->s.origin2[1] > 200) event->s.origin2[1] = 200; if (event->s.origin2[2] < 50) event->s.origin2[2] = 50; // always some upwards else if (event->s.origin2[2] > 300) event->s.origin2[2] = 300; } void ThrowClientHead (edict_t *self, int damage) { vec3_t vd; char *gibname; gibname = "models/objects/gibs/gib1/gib1.md3"; self->s.skinnum = 0; self->s.origin[2] += 32; self->s.frame = 0; self->s.modelindex = trap_ModelIndex (gibname); VectorSet (self->r.mins, -16, -16, 0); VectorSet (self->r.maxs, 16, 16, 16); self->takedamage = DAMAGE_NO; self->r.solid = SOLID_NOT; self->s.type = ET_GIB; self->s.sound = 0; self->s.effects = 0; self->flags |= FL_NO_KNOCKBACK; self->movetype = MOVETYPE_BOUNCE; VelocityForDamage (max(damage,50), vd); VectorAdd (self->velocity, vd, self->velocity); G_AddEvent( self, EV_GIB, 0, qfalse ); trap_LinkEntity( self ); } //================= //debris //================= void debris_die (edict_t *self, edict_t *inflictor, edict_t *attacker, int damage, vec3_t point) { G_FreeEdict (self); } void ThrowDebris (edict_t *self, char *modelname, float speed, vec3_t origin) { edict_t *chunk; vec3_t v; chunk = G_Spawn(); VectorCopy (origin, chunk->s.origin); chunk->s.modelindex = trap_ModelIndex (modelname); v[0] = 100 * crandom(); v[1] = 100 * crandom(); v[2] = 100 + 100 * crandom(); VectorMA (self->velocity, speed, v, chunk->velocity); chunk->movetype = MOVETYPE_BOUNCE; chunk->r.solid = SOLID_NOT; chunk->avelocity[0] = random()*600; chunk->avelocity[1] = random()*600; chunk->avelocity[2] = random()*600; chunk->think = G_FreeEdict; chunk->nextthink = level.timemsec + 5000 + random()*5000; chunk->s.frame = 0; chunk->flags = 0; chunk->classname = "debris"; chunk->takedamage = DAMAGE_YES; chunk->die = debris_die; trap_LinkEntity (chunk); } void BecomeExplosion1 (edict_t *self) { //flags are important if( self->item ) { if( self->item->type & IT_FLAG ) { G_Gametype_CTF_ResetFlag( self->s.team ); } } // turn entity into event G_TurnEntityIntoEvent ( self, EV_EXPLOSION1, 0 ); } void BecomeExplosion2 (edict_t *self) { // turn entity into event G_TurnEntityIntoEvent ( self, EV_EXPLOSION2, 0 ); } /*QUAKED path_corner (.5 .3 0) (-8 -8 -8) (8 8 8) TELEPORT Path corner entity that func_trains can be made to follow. -------- KEYS -------- target : point to next path_corner in the path. targetname : the train following the path or the previous path_corner in the path points to this. pathtarget: gets used when an entity that has this path_corner targeted touches it speed : speed of func_train while moving to the next path corner. This will override the speed value of the train. wait : number of seconds func_train will pause on path corner before moving to next path corner (default 0 - see Notes). notsingle : when set to 1, entity will not spawn in Single Player mode notfree : when set to 1, entity will not spawn in "Free for all" and "Tournament" modes. notduel : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. (jaltodo) notteam : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. notctf : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. (jaltodo) -------- SPAWNFLAGS -------- TELEPORT : &1 instant move to next target -------- NOTES -------- Setting the wait key to -1 will not make the train stop on the path corner, it will simply default to 0.*/ void path_corner_touch( edict_t *self, edict_t *other, cplane_t *plane, int surfFlags ) { vec3_t v; edict_t *next; if( other->movetarget != self ) return; if( other->enemy ) return; if( self->pathtarget ) { char *savetarget; savetarget = self->target; self->target = self->pathtarget; G_UseTargets( self, other ); self->target = savetarget; } if( self->target ) next = G_PickTarget( self->target ); else next = NULL; if( next && (next->spawnflags & 1) ) { VectorCopy( next->s.origin, v ); v[2] += next->r.mins[2]; v[2] -= other->r.mins[2]; VectorCopy (v, other->s.origin); next = G_PickTarget( next->target ); G_AddEvent( other, EV_TELEPORT, 0, qtrue ); } other->goalentity = other->movetarget = next; VectorSubtract (other->goalentity->s.origin, other->s.origin, v); other->ideal_yaw = vectoyaw (v); } void SP_path_corner(edict_t *self) { if (!self->targetname) { if (developer->integer) G_Printf ("path_corner with no targetname at %s\n", vtos(self->s.origin)); G_FreeEdict (self); return; } self->r.solid = SOLID_TRIGGER; self->touch = path_corner_touch; VectorSet (self->r.mins, -8, -8, -8); VectorSet (self->r.maxs, 8, 8, 8); self->r.svflags |= SVF_NOCLIENT; trap_LinkEntity (self); } /*QUAKED viewthing (0 .5 .8) (-8 -8 -8) (8 8 8) Just for the debugging level. Don't use */ static int robotron[4]; void TH_viewthing(edict_t *ent) { ent->s.frame = (ent->s.frame + 1) % 7; ent->nextthink = level.timemsec + game.framemsec; if (ent->spawnflags) { if (ent->s.frame == 0) { ent->spawnflags = (ent->spawnflags + 1) % 4 + 1; ent->s.modelindex = robotron[ent->spawnflags - 1]; } } } /*QUAKED info_null (0 .5 0) (-8 -8 -8) (8 8 8) Used as a positional target for light entities to create a spotlight effect. removed during gameplay. -------- KEYS -------- targetname : must match the target key of entity that uses this for pointing. -------- NOTES -------- A target_position can be used instead of this but was kept in for legacy purposes.*/ void SP_info_null (edict_t *self) { G_FreeEdict (self); } /*QUAKED info_notnull (0 .5 0) (-8 -8 -8) (8 8 8) Used as a positional target for entities that can use directional pointing. Kept during gameplay. -------- KEYS -------- targetname : must match the target key of entity that uses this for pointing. notsingle : when set to 1, entity will not spawn in Single Player mode notfree : when set to 1, entity will not spawn in "Free for all" and "Tournament" modes. notduel : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. (jaltodo) notteam : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. notctf : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. (jaltodo) -------- NOTES -------- A target_position can be used instead of this but was kept in for legacy purposes.*/ void SP_info_notnull (edict_t *self) { VectorCopy (self->s.origin, self->r.absmin); VectorCopy (self->s.origin, self->r.absmax); } /*QUAKED info_camp (0 .5 0) (-8 -8 -8) (8 8 8) Don't use. Can be used as pointing origin, but there are other entities for it. removed during gameplay. -------- KEYS -------- targetname : must match the target key of entity that uses this for pointing. notsingle : when set to 1, entity will not spawn in Single Player mode notfree : when set to 1, entity will not spawn in "Free for all" and "Tournament" modes. notduel : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. (jaltodo) notteam : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. notctf : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. (jaltodo) -------- NOTES -------- It was created to mark camp spots for monsters and bots, but it isn't used anymore and is only kept in for legacy purposes.*/ void SP_info_camp( edict_t *self ) { } /*QUAKED light (0 1 0) (-8 -8 -8) (8 8 8) START_OFF Non-displayed light. Default light value is 300. Default style is 0. If targeted, will toggle between on and off. Default _cone value is 10 (used to set size of light for spotlights) */ #define START_OFF 1 static void light_use( edict_t *self, edict_t *other, edict_t *activator ) { if( self->spawnflags & START_OFF) { trap_ConfigString( CS_LIGHTS+self->style, "m" ); self->spawnflags &= ~START_OFF; } else { trap_ConfigString( CS_LIGHTS+self->style, "a" ); self->spawnflags |= START_OFF; } } void SP_light( edict_t *self ) { if( !self->targetname/* || deathmatch->value*/) { G_FreeEdict( self ); return; } if( self->style >= 32 ) { self->use = light_use; if( self->spawnflags & START_OFF ) trap_ConfigString( CS_LIGHTS+self->style, "a" ); else trap_ConfigString( CS_LIGHTS+self->style, "m" ); } } //======================================================== // // FUNC_* // //======================================================== /*QUAKED func_group (0 0 0) ? Used to group brushes together just for editor convenience. */ //=========================================================== /*QUAKED func_wall (0 .5 .8) ? TRIGGER_SPAWN TOGGLE START_ON - - This is just a solid wall if not inhibited. Can be used for conditional walls. -------- KEYS -------- target : activate entities with this targetname when used targetname : use this name to target me notsingle : when set to 1, entity will not spawn in Single Player mode notfree : when set to 1, entity will not spawn in "Free for all" and "Tournament" modes. notduel : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. (jaltodo) notteam : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. notctf : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. (jaltodo) -------- SPAWNFLAGS -------- TRIGGER_SPAWN : &1 the wall will not be present until triggered it will then blink in to existance; it will kill anything that was in its way TOGGLE : &2 only valid for TRIGGER_SPAWN walls this allows the wall to be turned on and off START_ON : &4 only valid for TRIGGER_SPAWN walls the wall will initially be present -------- NOTES -------- Untested entity*/ void func_wall_use( edict_t *self, edict_t *other, edict_t *activator ) { if( self->r.solid == SOLID_NOT ) { self->r.solid = SOLID_BSP; self->r.svflags &= ~SVF_NOCLIENT; KillBox (self); } else { self->r.solid = SOLID_NOT; self->r.svflags |= SVF_NOCLIENT; } trap_LinkEntity( self ); if( !(self->spawnflags & 2) ) self->use = NULL; } void SP_func_wall( edict_t *self ) { self->movetype = MOVETYPE_PUSH; trap_SetBrushModel( self, self->model ); // just a wall if ((self->spawnflags & 7) == 0) { self->r.solid = SOLID_BSP; trap_LinkEntity (self); return; } // it must be TRIGGER_SPAWN if (!(self->spawnflags & 1)) { // G_Printf ("func_wall missing TRIGGER_SPAWN\n"); self->spawnflags |= 1; } // yell if the spawnflags are odd if (self->spawnflags & 4) { if (!(self->spawnflags & 2)) { if (developer->integer) G_Printf ("func_wall START_ON without TOGGLE\n"); self->spawnflags |= 2; } } self->use = func_wall_use; if (self->spawnflags & 4) { self->r.solid = SOLID_BSP; } else { self->r.solid = SOLID_NOT; self->r.svflags |= SVF_NOCLIENT; } trap_LinkEntity (self); } //=========================================================== /*QUAKED func_static (0 .5 .8) ? Static non-solid bspmodel. Can be used for conditional walls and models. -------- KEYS -------- model2 : path/name of model to include (eg: models/mapobjects/bitch/fembotbig.md3). origin : alternate method of setting XYZ origin of .md3 model included with entity (See Notes). light : constantLight radius of .md3 model included with entity. Has no effect on the entity's brushes (default 0). color : constantLight color of .md3 model included with entity. Has no effect on the entity's brushes (default 1 1 1). targetname : NOT SUPPORTED BY RENDERER - if set, a func_button or trigger can make entity disappear from the game (See Notes). notsingle : when set to 1, entity will not spawn in Single Player mode notfree : when set to 1, entity will not spawn in "Free for all" and "Tournament" modes. notduel : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. (jaltodo) notteam : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. notctf : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. (jaltodo) -------- NOTES -------- When using the model2 key, the origin point of the model will correspond to the origin point defined by either the origin brush or the origin coordinate value. If a model is included with a targeted func_static, the brush(es) of the entity will be removed from the game but the .md3 model won't: it will automatically be moved to the (0 0 0) world origin so you should NOT include an .md3 model to a targeted func_static.*/ void SP_func_static (edict_t *ent) { G_InitMover ( ent ); ent->movetype = MOVETYPE_NONE; trap_LinkEntity (ent); } //=========================================================== /*QUAKED func_object (0 .5 .8) ? TRIGGER_SPAWN - - This is solid bmodel that will fall if its support it removed. -------- KEYS -------- origin : alternate method of setting XYZ origin of .md3 model included with entity (See Notes). light : constantLight radius of .md3 model included with entity. Has no effect on the entity's brushes (default 0). color : constantLight color of .md3 model included with entity. Has no effect on the entity's brushes (default 1 1 1). target : fire targets with this name at being used. targetname : name to be targeted with. notsingle : when set to 1, entity will not spawn in Single Player mode notfree : when set to 1, entity will not spawn in "Free for all" and "Tournament" modes. notduel : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. (jaltodo) notteam : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. notctf : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. (jaltodo) -------- SPAWNFLAGS -------- TRIGGER_SPAWN : &1 spawn this entity when target is fired -------- NOTES -------- model2 is not supported in func_objects, only map brushes can be safely used as model*/ void func_object_touch( edict_t *self, edict_t *other, cplane_t *plane, int surfFlags ) { // only squash thing we fall on top of if( !plane ) return; if( plane->normal[2] < 1.0 ) return; if( other->takedamage == DAMAGE_NO ) return; T_Damage( other, self, self, vec3_origin, self->s.origin, vec3_origin, self->dmg, 1, 0, MOD_CRUSH ); } void func_object_release( edict_t *self ) { self->movetype = MOVETYPE_TOSS; self->touch = func_object_touch; } void func_object_use( edict_t *self, edict_t *other, edict_t *activator ) { self->r.solid = SOLID_BSP; self->r.svflags &= ~SVF_NOCLIENT; self->use = NULL; KillBox( self ); func_object_release( self ); } void SP_func_object( edict_t *self ) { trap_SetBrushModel( self, self->model ); self->r.mins[0] += 1; self->r.mins[1] += 1; self->r.mins[2] += 1; self->r.maxs[0] -= 1; self->r.maxs[1] -= 1; self->r.maxs[2] -= 1; if ( !self->dmg ) self->dmg = 100; if( self->spawnflags == 0 ) { self->r.solid = SOLID_BSP; self->movetype = MOVETYPE_PUSH; self->think = func_object_release; self->nextthink = level.timemsec + 2 * game.framemsec; } else { self->r.solid = SOLID_NOT; self->movetype = MOVETYPE_PUSH; self->use = func_object_use; self->r.svflags |= SVF_NOCLIENT; } self->r.clipmask = MASK_MONSTERSOLID; trap_LinkEntity( self ); } //=========================================================== /*QUAKED func_explosive (0 .5 .8) ? TRIGGER_SPAWN - - Any brush that you want to explode or break apart. If you want an ex0plosion, set dmg and it will do a radius explosion of that amount at the center of the bursh. If targeted it will not be shootable. -------- KEYS -------- health : defaults to 100. mass : defaults to 75. This determines how much debris is emitted when it explodes. You get one large chunk per 100 of mass (up to 8) and one small chunk per 25 of mass (up to 16). So 800 gives the most. model2 : a md3 model. origin : alternate method of setting XYZ origin of .md3 model included with entity (See Notes). light : constantLight radius of .md3 model included with entity. Has no effect on the entity's brushes (default 0). color : constantLight color of .md3 model included with entity. Has no effect on the entity's brushes (default 1 1 1). target : fire targets with this name at being used. targetname : name to be targeted with. notsingle : when set to 1, entity will not spawn in Single Player mode notfree : when set to 1, entity will not spawn in "Free for all" and "Tournament" modes. notduel : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. (jaltodo) notteam : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. notctf : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. (jaltodo) -------- SPAWNFLAGS -------- TRIGGER_SPAWN : &1 spawn this entity when target is fired -------- NOTES -------- Untested: model2 models might not collide perfectly if used with a brush origin*/ void func_explosive_explode( edict_t *self, edict_t *inflictor, edict_t *attacker, int damage, vec3_t point ) { vec3_t origin; vec3_t chunkorigin; vec3_t size; int count; int mass; // bmodel origins are (0 0 0), we need to adjust that here VectorScale( self->r.size, 0.5, size ); VectorAdd( self->r.absmin, size, origin ); VectorCopy( origin, self->s.origin ); self->takedamage = DAMAGE_NO; self->s.light = 0; if( self->dmg ) T_RadiusDamage( self, attacker, NULL, self->dmg, self->dmg, 0, NULL, self->dmg+40, MOD_EXPLOSIVE ); VectorSubtract( self->s.origin, inflictor->s.origin, self->velocity ); VectorNormalize( self->velocity ); VectorScale( self->velocity, 150, self->velocity ); // start chunks towards the center VectorScale( size, 0.5, size ); mass = self->mass; if( !mass ) mass = 75; // big chunks if( mass >= 100 ) { count = mass / 100; if( count > 8 ) count = 8; while( count-- ) { chunkorigin[0] = origin[0] + crandom() * size[0]; chunkorigin[1] = origin[1] + crandom() * size[1]; chunkorigin[2] = origin[2] + crandom() * size[2]; ThrowDebris( self, "models/objects/debris1/tris.md2", 1, chunkorigin ); } } // small chunks count = mass / 25; if( count > 16 ) count = 16; if( count < 1 ) count = 1; while( count-- ) { chunkorigin[0] = origin[0] + crandom() * size[0]; chunkorigin[1] = origin[1] + crandom() * size[1]; chunkorigin[2] = origin[2] + crandom() * size[2]; ThrowDebris( self, "models/objects/debris2/tris.md2", 2, chunkorigin ); } G_UseTargets( self, attacker ); if( self->dmg ) BecomeExplosion1( self ); else G_FreeEdict( self ); } void func_explosive_use( edict_t *self, edict_t *other, edict_t *activator ) { func_explosive_explode( self, self, other, ceil(self->health), vec3_origin ); } void func_explosive_spawn( edict_t *self, edict_t *other, edict_t *activator ) { self->r.solid = SOLID_BSP; self->r.svflags &= ~SVF_NOCLIENT; self->use = NULL; KillBox( self ); trap_LinkEntity( self ); } void SP_func_explosive( edict_t *self ) { //newgametypes[start] Allow func explosive in multiplayer, why not? //if (deathmatch->integer) //{ // auto-remove for deathmatch // G_FreeEdict (self); // return; //}//newgametypes[end] trap_ModelIndex( "models/objects/debris1/tris.md2" ); trap_ModelIndex( "models/objects/debris2/tris.md2" ); G_InitMover( self ); if( self->spawnflags & 1 ) { self->r.svflags |= SVF_NOCLIENT; self->r.solid = SOLID_NOT; self->use = func_explosive_spawn; } else { if( self->targetname ) self->use = func_explosive_use; } if( self->use != func_explosive_use ) { if( !self->health ) self->health = 100; self->die = func_explosive_explode; self->takedamage = DAMAGE_YES; } trap_LinkEntity( self ); } //======================================================== // // MISC_* // //======================================================== /*QUAKED light_mine (0 1 0) (-2 -2 -12) (2 2 12) */ void SP_light_mine( edict_t *ent ) { ent->movetype = MOVETYPE_NONE; ent->r.solid = SOLID_BBOX; ent->s.modelindex = trap_ModelIndex( "models/objects/minelite/light1/tris.md2" ); trap_LinkEntity( ent ); } //===================================================== /*QUAKED misc_teleporter_dest (1 .5 .25) (-32 -32 -24) (32 32 -16) Teleport destination location point for trigger_teleporter entities. -------- KEYS -------- angle : direction in which player will look when teleported. targetname : make the trigger_teleporter point to this. notsingle : when set to 1, entity will not spawn in Single Player mode notfree : when set to 1, entity will not spawn in "Free for all" and "Tournament" modes. notduel : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. (jaltodo) notteam : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. notctf : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. (jaltodo) -------- NOTES -------- Does the same thing as info_teleport_destination*/ void SP_misc_teleporter_dest( edict_t *ent ) { } //===================================================== /*QUAKED misc_model (1 .5 .25) (-16 -16 -16) (16 16 16) Generic placeholder for inserting .md3 models in game. Requires compilation of map geometry to be added to level. -------- KEYS -------- angle: direction in which model will be oriented. model : path/name of model to use (eg: models/mapobjects/teleporter/teleporter.md3).*/ void SP_misc_model (edict_t *ent) { G_FreeEdict (ent); } //=========================================================== /*QUAKED misc_portal_surface (1 .5 .25) (-8 -8 -8) (8 8 8) The portal surface nearest this entity will show a view from the targeted misc_portal_camera, or a mirror view if untargeted. This must be within 64 world units of the surface! -------- KEYS -------- notsingle : when set to 1, entity will not spawn in Single Player mode notfree : when set to 1, entity will not spawn in "Free for all" and "Tournament" modes. notduel : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. (jaltodo) notteam : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. notctf : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. (jaltodo) -------- NOTES -------- The entity must be no farther than 64 units away from the portal surface to lock onto it. To make a mirror, apply the common/mirror shader to the surface, place this entity near it but don't target a misc_portal_camera.*/ void misc_portal_surface_think( edict_t *ent ) { if( !ent->r.owner || !ent->r.owner->r.inuse ) VectorCopy( ent->s.origin, ent->s.origin2 ); else VectorCopy( ent->r.owner->s.origin, ent->s.origin2 ); ent->nextthink = level.timemsec + game.framemsec; } void locateCamera( edict_t *ent ) { vec3_t dir; edict_t *target; edict_t *owner; owner = G_PickTarget( ent->target ); if( !owner ) { G_Printf( "Couldn't find target for %s\n", ent->classname ); G_FreeEdict(ent); return; } // modelindex2 holds the rotate speed if( owner->spawnflags & 1 ) ent->s.modelindex2 = 25; else if ( owner->spawnflags & 2 ) ent->s.modelindex2 = 75; // swing camera ? if( owner->spawnflags & 4 ) // set to 0 for no rotation at all ent->s.effects &= ~EF_ROTATE_AND_BOB; else ent->s.effects |= EF_ROTATE_AND_BOB; ent->r.owner = owner; ent->think = misc_portal_surface_think; ent->nextthink = level.timemsec + game.framemsec; // see if the portal_camera has a target if( owner->target ) target = G_PickTarget( owner->target ); else target = NULL; if( target ) { VectorSubtract( target->s.origin, owner->s.origin, dir ); VectorNormalize( dir ); } else { G_SetMovedir( owner->s.angles, dir ); } ent->s.skinnum = DirToByte( dir ); ent->s.frame = owner->count; } void SP_misc_portal_surface( edict_t *ent ) { VectorClear( ent->r.mins ); VectorClear( ent->r.maxs ); trap_LinkEntity( ent ); ent->s.type = ET_PORTALSURFACE; ent->s.modelindex = 1; ent->r.svflags = SVF_PORTAL|SVF_FORCEOLDORIGIN; // mirror if( !ent->target ) { ent->think = misc_portal_surface_think; ent->nextthink = level.timemsec + game.framemsec; } else { ent->think = locateCamera; ent->nextthink = level.timemsec + 1000; } } //=========================================================== /*QUAKED misc_portal_camera (1 .5 .25) (-8 -8 -8) (8 8 8) SLOWROTATE FASTROTATE NOROTATE Portal camera. This camera is used to project its view onto a portal surface in the level through the intermediary of a misc_portal_surface entity. Use the "angles" key or target a target_position or info_notnull entity to set the camera's pointing direction. -------- KEYS -------- angles: this sets the pitch and yaw aiming angles of the portal camera (default 0 0). Use "roll" key to set roll angle. target : point this to a target_position entity to set the camera's pointing direction. targetname : a misc_portal_surface portal surface indicator must point to this. roll: roll angle of camera. A value of 0 is upside down and 180 is the same as the player's view. notsingle : when set to 1, entity will not spawn in Single Player mode notfree : when set to 1, entity will not spawn in "Free for all" and "Tournament" modes. notduel : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. (jaltodo) notteam : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. notctf : when set to 1, entity will not spawn in "Teamplay" and "CTF" modes. (jaltodo) -------- SPAWNFLAGS -------- SLOWROTATE : makes the portal camera rotate slowly along the roll axis. FASTROTATE : makes the portal camera rotate faster along the roll axis. NOROTATE : disables rotation -------- NOTES -------- Both the setting "angles" key or "targeting a target_position" methods can be used to aim the camera. However, the target_position method is simpler. In both cases, the "roll" key must be used to set the roll angle. If either the SLOWROTATE or FASTROTATE spawnflag is set, then the "roll" value is irrelevant.*/ void SP_misc_portal_camera(edict_t *ent) { VectorClear( ent->r.mins ); VectorClear( ent->r.maxs ); trap_LinkEntity( ent ); ent->r.svflags = SVF_NOCLIENT; ent->count = (int)(st.roll / 360.0f * 256.0f); }