mirror of
https://github.com/blendogames/thirtyflightsofloving.git
synced 2024-11-14 16:40:57 +00:00
3ce18b138c
Simplified loading and saving of cvars in slider menu control. Enlarged text in Game, Multiplayer, and Options menus. Fixed repeat of failed file causing HTTP downloads to restart. Added cl_zoommode cvar to simplify Lazarus zoom command. Changed zoom command to use new cl_zoommode cvar in default Lazarus and missionpack DLLs. Removed unused "crossh" cvar in default Lazarus and missionpack DLLs. Fixed Makron not having a classname when spawned from Jorg in default Lazarus and missionpack DLLs. Made Tactician Gunner ignore small amounts of damage in missionpack DLL.
358 lines
13 KiB
C
358 lines
13 KiB
C
/*
|
|
===========================================================================
|
|
Copyright (C) 1997-2001 Id Software, Inc.
|
|
Copyright (C) 2000-2002 Mr. Hyde and Mad Dog
|
|
|
|
This file is part of Lazarus Quake 2 Mod source code.
|
|
|
|
Lazarus Quake 2 Mod source code 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.
|
|
|
|
Lazarus Quake 2 Mod source code 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 Lazarus Quake 2 Mod source code; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
===========================================================================
|
|
*/
|
|
|
|
#include "g_local.h"
|
|
|
|
// Lazarus: removed all waterlevel-dependent stuff, which I don't get, and
|
|
// the fake crosshair, which looked... really bad
|
|
|
|
cvar_t *tpp;
|
|
cvar_t *tpp_auto;
|
|
//cvar_t *crossh;
|
|
|
|
void ChasecamTrack (edict_t *ent);
|
|
|
|
/* The ent is the owner of the chasecam */
|
|
void ChasecamStart (edict_t *ent)
|
|
{
|
|
/* This creates a tempory entity we can manipulate within this
|
|
* function */
|
|
edict_t *chasecam;
|
|
/* Don't work on a spectator! */
|
|
if (ent->client->resp.spectator)
|
|
return;
|
|
//Don't turn back on during intermission!
|
|
if (level.intermissiontime)
|
|
return;
|
|
/* Tell everything that looks at the toggle that our chasecam is on
|
|
* and working */
|
|
ent->client->chasetoggle = 1;
|
|
/* Make our gun model "non-existent" so it's more realistic to the
|
|
* player using the chasecam */
|
|
ent->client->ps.gunindex = 0;
|
|
chasecam = G_Spawn ();
|
|
chasecam->owner = ent;
|
|
chasecam->solid = SOLID_NOT;
|
|
chasecam->movetype = MOVETYPE_FLYMISSILE;
|
|
ent->client->ps.pmove.pm_flags |= PMF_NO_PREDICTION; // this turns off Quake2's inclination to predict where the camera is going,
|
|
// making a much smoother ride
|
|
ent->svflags |= SVF_NOCLIENT; // this line tells Quake2 not to send the unnecessary info about the camera to other players
|
|
/* Now, make the angles of the player model, (!NOT THE HUMAN VIEW!) be
|
|
* copied to the same angle of the chasecam entity */
|
|
VectorCopy (ent->s.angles, chasecam->s.angles);
|
|
/* Clear the size of the entity, so it DOES technically have a size,
|
|
* but that of '0 0 0'-'0 0 0'. (xyz, xyz). mins = Minimum size,
|
|
* maxs = Maximum size */
|
|
VectorClear (chasecam->mins);
|
|
VectorClear (chasecam->maxs);
|
|
/* Make the chasecam's origin (position) be the same as the player
|
|
* entity's because as the camera starts, it will force itself out
|
|
* slowly backwards from the player model */
|
|
VectorCopy (ent->s.origin, chasecam->s.origin);
|
|
chasecam->classname = "chasecam";
|
|
chasecam->prethink = ChasecamTrack;
|
|
// Lazarus: Need think???
|
|
chasecam->think = ChasecamTrack;
|
|
ent->client->chasecam = chasecam;
|
|
ent->client->oldplayer = G_Spawn();
|
|
CheckChasecam_Viewent(ent);
|
|
// MakeFakeCrosshair (ent);
|
|
|
|
// remove reflection of real player, if any
|
|
DeleteReflection (ent, -1);
|
|
}
|
|
|
|
/* ent = chasecam */
|
|
void ChasecamRestart (edict_t *ent)
|
|
{
|
|
/* Keep thinking this function to check all the time whether the
|
|
* player is out of the water */
|
|
/* If the player is dead, the camera is not wanted... Kill me and stop
|
|
* the function. (return;) */
|
|
if (ent->owner->health <= 0)
|
|
{
|
|
G_FreeEdict(ent);
|
|
return;
|
|
}
|
|
/* If the player is still completly underwater, break the routine
|
|
unless tpp has changed!*/
|
|
// if (ent->owner->waterlevel && !tpp->value)
|
|
// return;
|
|
//Put camera back
|
|
ChasecamStart (ent->owner);
|
|
//Remove this temporary ent
|
|
G_FreeEdict (ent);
|
|
}
|
|
|
|
/* Here, the "ent" is referring to the client, the player that owns the
|
|
* chasecam, and the "opt" integer is telling the function whether to
|
|
* totally get rid of the camera, or to put it into the background while
|
|
* it checks if the player is out of the water or not. */
|
|
void ChasecamRemove (edict_t *ent, int opt)
|
|
{
|
|
/* Stop the chasecam from moving */
|
|
VectorClear (ent->client->chasecam->velocity);
|
|
/* Make the weapon model of the player appear on screen for 1st
|
|
* person reality and aiming */
|
|
//Don't turn back on during intermission!
|
|
if (!level.intermissiontime)
|
|
ent->client->ps.gunindex = gi.modelindex(ent->client->pers.weapon->view_model);
|
|
/* Make our invisible appearance the same model as the display entity
|
|
* that mimics us while in chasecam mode */
|
|
ent->s.modelindex = ent->client->oldplayer->s.modelindex;
|
|
ent->svflags &= ~SVF_NOCLIENT;
|
|
|
|
// DestroyFakeCrosshair (ent);
|
|
|
|
if (opt == OPTION_BACKGROUND)
|
|
{
|
|
ent->client->chasetoggle = 0;
|
|
G_FreeEdict (ent->client->chasecam);
|
|
G_FreeEdict (ent->client->oldplayer);
|
|
ent->client->oldplayer = NULL;
|
|
ent->client->chasecam = G_Spawn ();
|
|
ent->client->chasecam->owner = ent;
|
|
ent->client->chasecam->solid = SOLID_NOT;
|
|
ent->client->chasecam->movetype = MOVETYPE_FLYMISSILE;
|
|
VectorClear (ent->client->chasecam->mins);
|
|
VectorClear (ent->client->chasecam->maxs);
|
|
ent->client->chasecam->classname = "chasecam";
|
|
ent->client->chasecam->prethink = ChasecamRestart; // begin checking for emergence from the water
|
|
// Lazarus: Need think don't we???
|
|
ent->client->chasecam->think = ChasecamRestart;
|
|
}
|
|
else if (opt == OPTION_OFF)
|
|
{
|
|
G_FreeEdict (ent->client->oldplayer);
|
|
ent->client->oldplayer = NULL;
|
|
ent->client->chasetoggle = 0;
|
|
G_FreeEdict (ent->client->chasecam);
|
|
ent->client->chasecam = NULL;
|
|
}
|
|
}
|
|
|
|
/* The "ent" is the chasecam */
|
|
void ChasecamTrack (edict_t *ent)
|
|
{
|
|
/* Create tempory vectors and trace variables */
|
|
trace_t tr;
|
|
vec3_t spot1, spot2, dir;
|
|
vec3_t forward, right, up,angles;
|
|
int distance;
|
|
double tot; //mxd. int -> double
|
|
|
|
ent->nextthink = level.time + 0.100;
|
|
/* if our owner is under water, run the remove routine to repeatedly
|
|
* check for emergment from water */
|
|
// if (ent->owner->waterlevel && !tpp->value)
|
|
// {
|
|
// ChasecamRemove (ent->owner, OPTION_BACKGROUND);
|
|
// return;
|
|
// }
|
|
/* get the CLIENT's angle, and break it down into direction vectors,
|
|
* of forward, right, and up. VERY useful */
|
|
VectorCopy(ent->owner->client->v_angle,angles);
|
|
if (angles[PITCH] > 56)
|
|
angles[PITCH] = 56;
|
|
AngleVectors (angles, forward, right, up);
|
|
VectorNormalize(forward);
|
|
/* go starting at the player's origin, forward, ent->chasedist1
|
|
* distance, and save the location in vector spot2 */
|
|
VectorMA (ent->owner->s.origin, -ent->chasedist1, forward, spot2);
|
|
/* make spot2 a bit higher, by adding viewheight to the Z coordinate */
|
|
spot2[2] += (ent->owner->viewheight + 16);
|
|
// jump animation lifts
|
|
if (!ent->owner->groundentity)
|
|
spot2[2] += 16;
|
|
/* make the tr traceline trace from the player model's position, to spot2,
|
|
* ignoring the player, with a mask. */
|
|
tr = gi.trace (ent->owner->s.origin, vec3_origin, vec3_origin, spot2, ent->owner, MASK_SOLID);
|
|
/* subtract the endpoint from the start point for length and
|
|
* direction manipulation */
|
|
VectorSubtract (tr.endpos, ent->owner->s.origin, spot1);
|
|
/* in this case, length */
|
|
ent->chasedist1 = VectorLength (spot1);
|
|
/* go, starting from the end of the trace, 2 points forward (client
|
|
* angles) and save the location in spot2 */
|
|
VectorMA (tr.endpos, 2, forward, spot2);
|
|
/* make spot1 the same for tempory vector modification and make spot1
|
|
* a bit higher than spot2 */
|
|
VectorCopy (spot2, spot1);
|
|
spot1[2] += 32;
|
|
/* another trace from spot2 to spot1, ignoring player, no masks */
|
|
tr = gi.trace (spot2, vec3_origin, vec3_origin, spot1, ent->owner, MASK_SOLID);
|
|
/* if we hit something, copy the trace end to spot2 and lower spot2 */
|
|
if (tr.fraction < 1.000)
|
|
{
|
|
VectorCopy (tr.endpos, spot2);
|
|
spot2[2] -= 32;
|
|
}
|
|
/* subtract endpos spot2 from startpos the camera origin, saving it to
|
|
* the dir vector, and normalize dir for a direction from the camera
|
|
* origin, to the spot2 */
|
|
VectorSubtract (spot2, ent->s.origin, dir);
|
|
distance = VectorLength (dir);
|
|
VectorNormalize (dir);
|
|
/* another traceline */
|
|
tr = gi.trace (ent->s.origin, vec3_origin, vec3_origin, spot2, ent->owner, MASK_SOLID);
|
|
/* if we DON'T hit anyting, do some freaky stuff */
|
|
if (tr.fraction == 1.000)
|
|
{
|
|
/* subtract the endpos camera position, from the startpos, the
|
|
* player, and save in spot1. Normalize spot1 for a direction, and
|
|
* make that direction the angles of the chasecam for copying to the
|
|
* clients view angle which is displayed to the client. (human) */
|
|
VectorSubtract (ent->s.origin, ent->owner->s.origin, spot1);
|
|
VectorNormalize (spot1);
|
|
VectorCopy (spot1, ent->s.angles);
|
|
/* calculate the percentages of the distances, and make sure we're
|
|
* not going too far, or too short, in relation to our panning
|
|
* speed of the chasecam entity */
|
|
tot = (distance * 0.400);
|
|
/* if we're going too fast, make us top speed */
|
|
if (tot > 5.200)
|
|
{
|
|
ent->velocity[0] = ((dir[0] * distance) * 5.2);
|
|
ent->velocity[1] = ((dir[1] * distance) * 5.2);
|
|
ent->velocity[2] = ((dir[2] * distance) * 5.2);
|
|
}
|
|
else
|
|
{
|
|
/* if we're NOT going top speed, but we're going faster than
|
|
* 1, relative to the total, make us as fast as we're going */
|
|
if (tot > 1.000)
|
|
{
|
|
ent->velocity[0] = ((dir[0] * distance) * tot);
|
|
ent->velocity[1] = ((dir[1] * distance) * tot);
|
|
ent->velocity[2] = ((dir[2] * distance) * tot);
|
|
}
|
|
else
|
|
{
|
|
/* if we're not going faster than one, don't accelerate our
|
|
* speed at all, make us go slow to our destination */
|
|
ent->velocity[0] = (dir[0] * distance);
|
|
ent->velocity[1] = (dir[1] * distance);
|
|
ent->velocity[2] = (dir[2] * distance);
|
|
}
|
|
}
|
|
/* subtract endpos,player position, from chasecam position to get
|
|
* a length to determine whether we should accelerate faster from
|
|
* the player or not */
|
|
VectorSubtract (ent->owner->s.origin, ent->s.origin, spot1);
|
|
if (VectorLength(spot1) < 20)
|
|
{
|
|
ent->velocity[0] *= 2;
|
|
ent->velocity[1] *= 2;
|
|
ent->velocity[2] *= 2;
|
|
}
|
|
}
|
|
/* if we DID hit something in the tr.fraction call ages back, then
|
|
* make the spot2 we created, the position for the chasecamera. */
|
|
else
|
|
VectorCopy (spot2, ent->s.origin);
|
|
/* add to the distance between the player and the camera */
|
|
ent->chasedist1 += 2;
|
|
/* if we're too far away, give us a maximum distance */
|
|
if (ent->chasedist1 > (60.00 + ent->owner->client->zoom))
|
|
ent->chasedist1 = (60.00 + ent->owner->client->zoom);
|
|
/* if we haven't gone anywhere since the last think routine, and we
|
|
* are greater than 20 points in the distance calculated, add one to
|
|
* the second chasedistance variable
|
|
* The "ent->movedir" is a vector which is not used in this entity, so
|
|
* we can use this a tempory vector belonging to the chasecam, which
|
|
* can be carried through think routines. */
|
|
if (VectorCompare(ent->movedir, ent->s.origin))
|
|
{
|
|
if (distance > 20)
|
|
ent->chasedist2++;
|
|
}
|
|
/* if we've buggered up more than 3 times, there must be some mistake,
|
|
* so restart the camera so we re-create a chasecam, destroy the old one,
|
|
* slowly go outwards from the player, and keep thinking this routing in
|
|
* the new camera entity */
|
|
if (ent->chasedist2 > 3)
|
|
{
|
|
G_FreeEdict (ent->owner->client->oldplayer);
|
|
ChasecamStart (ent->owner);
|
|
G_FreeEdict(ent);
|
|
return;
|
|
}
|
|
/* Copy the position of the chasecam now, and stick it to the movedir
|
|
* variable, for position checking when we rethink this function */
|
|
VectorCopy (ent->s.origin, ent->movedir);
|
|
/* MUST LINK SINCE WE CHANGED THE ORIGIN! */
|
|
gi.linkentity (ent);
|
|
// UpdateFakeCrosshair (ent->owner);
|
|
}
|
|
|
|
void Cmd_Chasecam_Toggle (edict_t *ent)
|
|
{
|
|
// Lazarus: Don't allow thirdperson when using spycam
|
|
if (!ent->deadflag && !ent->client->spycam)
|
|
{
|
|
if (ent->client->chasetoggle)
|
|
ChasecamRemove (ent, OPTION_OFF);
|
|
// Knightmare- don't use server chasecam if client chasecam is on
|
|
#ifdef KMQUAKE2_ENGINE_MOD
|
|
else if (!cl_thirdperson->value || deathmatch->value || coop->value)
|
|
#else
|
|
else
|
|
#endif
|
|
ChasecamStart (ent);
|
|
}
|
|
}
|
|
|
|
void CheckChasecam_Viewent (edict_t *ent)
|
|
{
|
|
vec3_t angles;
|
|
/*
|
|
Oldplayer is the fake player that everyone else sees.
|
|
Assign the same client as the ent we're following so the game
|
|
can read any vars it wants from there
|
|
*/
|
|
if (!ent->client->oldplayer->client)
|
|
ent->client->oldplayer->client = ent->client;
|
|
/* Copy the angle and model from ourselves to the old player.
|
|
Even though people can't see us we still have all this stuff */
|
|
if ((ent->client->chasetoggle == 1) && (ent->client->oldplayer))
|
|
{
|
|
// Lazarus
|
|
// if (ent->client->use)
|
|
// VectorCopy (ent->client->oldplayer->s.angles, angles);
|
|
// ent->client->oldplayer->s = ent->s; //Copy player related info
|
|
// if (ent->client->use)
|
|
// VectorCopy (angles, ent->client->oldplayer->s.angles);
|
|
|
|
if (ent->client->use && !ent->client->push)
|
|
VectorCopy (ent->client->oldplayer->s.angles, angles);
|
|
ent->client->oldplayer->s = ent->s; //Copy player related info
|
|
// Lazarus: s.numbers shouldn't be the same
|
|
ent->client->oldplayer->s.number = ent->client->oldplayer - g_edicts;
|
|
if (ent->client->use && !ent->client->push)
|
|
VectorCopy (angles, ent->client->oldplayer->s.angles);
|
|
ent->client->oldplayer->flags = ent->flags;
|
|
// end Lazarus
|
|
gi.linkentity (ent->client->oldplayer);
|
|
}
|
|
}
|
|
|