2447 lines
59 KiB
C
2447 lines
59 KiB
C
#include "g_Physics.h"
|
|
#include "g_local.h"
|
|
|
|
#include "Vector.h"
|
|
#include "Angles.h"
|
|
#include "PrimitiveDisplayHack.h"
|
|
#include "SinglyLinkedList.h"
|
|
#include "q_Physics.h"
|
|
#include "g_BoundingForm.h"
|
|
#include "Utilities.h"
|
|
#include "p_main.h"
|
|
#include "random.h"
|
|
#include "p_anim_branch.h"
|
|
#include "p_anims.h"
|
|
#include "fx.h"
|
|
#include "g_playstats.h"
|
|
#include "p_actions.h"
|
|
|
|
static void Physics_None(edict_t *self);
|
|
static void Physics_Static(edict_t *self);
|
|
static void Physics_NoclipMove(edict_t *self);
|
|
static void Physics_FlyMove(edict_t *self);
|
|
static void Physics_StepMove(edict_t *self);
|
|
static void Physics_Push(edict_t *self);
|
|
static void Physics_ScriptAngular(edict_t *self);
|
|
|
|
void (*physicsFuncs[NUM_PHYSICSTYPES])(edict_t *self) =
|
|
{
|
|
Physics_None, // PHYSICSTYPE_NONE
|
|
Physics_Static, // PHYSICSTYPE_STATIC
|
|
Physics_NoclipMove, // PHYSICSTYPE_NOCLIP
|
|
Physics_FlyMove, // PHYSICSTYPE_FLY
|
|
Physics_StepMove, // PHYSICSTYPE_STEP
|
|
Physics_Push, // PHYSICSTYPE_PUSH
|
|
Physics_Push, // PHYSICSTYPE_STOP
|
|
Physics_FlyMove, // MOVETYPE_FLYMISSILE
|
|
Physics_ScriptAngular, // PHYSICSTYPE_SCRIPT_ANGULAR
|
|
};
|
|
|
|
void PhysicsCheckWaterTransition(edict_t *self)
|
|
{//fixme: a high detail option? Or just not in netplay?- maybe a flag for client to take care of
|
|
//disabling for now since it might cause too much net traffic
|
|
qboolean wasinwater, isinwater;
|
|
trace_t trace;
|
|
int size;
|
|
|
|
if(deathmatch->value || coop->value)
|
|
return;
|
|
|
|
// check for water transition
|
|
wasinwater = (self->watertype & MASK_WATER);
|
|
self->watertype = gi.pointcontents (self->s.origin);
|
|
isinwater = self->watertype & MASK_WATER;
|
|
|
|
if (!wasinwater && isinwater)
|
|
{
|
|
gi.trace(self->s.old_origin, vec3_origin, vec3_origin, self->s.origin, self, MASK_WATER,&trace);
|
|
}
|
|
else if (wasinwater && !isinwater)
|
|
{
|
|
gi.trace(self->s.origin, vec3_origin, vec3_origin, self->s.old_origin, self, MASK_WATER,&trace);
|
|
}
|
|
else
|
|
return;
|
|
|
|
if(trace.fraction==1.0)
|
|
return;
|
|
|
|
//fixme: just put a flag on them and do the effect on the other side?
|
|
size = ceil(VectorLength(self->size) + VectorLength(self->velocity)/10);
|
|
if(size<10)
|
|
size = 10;
|
|
else if(size>255)
|
|
size = 255;
|
|
|
|
gi.CreateEffect(NULL,
|
|
FX_WATER_ENTRYSPLASH,
|
|
CEF_FLAG6 | CEF_FLAG7,
|
|
trace.endpos,
|
|
"bd",
|
|
size,
|
|
trace.plane.normal);
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------
|
|
//---------------------------------------------------------------------------------
|
|
static void Physics_None(edict_t *self)
|
|
{
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------
|
|
//---------------------------------------------------------------------------------
|
|
static void Physics_Static(edict_t *self)
|
|
{
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------
|
|
//---------------------------------------------------------------------------------
|
|
static void Physics_NoclipMove(edict_t *self)
|
|
{
|
|
VectorMA(self->s.angles, FRAMETIME, self->avelocity, self->s.angles);
|
|
VectorMA(self->s.origin, FRAMETIME, self->velocity, self->s.origin);
|
|
|
|
gi.linkentity(self);
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------
|
|
//---------------------------------------------------------------------------------
|
|
static void Physics_FlyMove(edict_t *self)
|
|
{
|
|
FormMove_t formMove;
|
|
|
|
if(self->physicsFlags & PF_RESIZE)
|
|
{
|
|
if(gi.ResizeBoundingForm(self, &formMove))
|
|
{
|
|
self->physicsFlags &= ~PF_RESIZE;
|
|
}
|
|
else
|
|
{
|
|
return; // if an ent can't be resized, then it probably can't be moved either
|
|
}
|
|
}
|
|
|
|
// update angles
|
|
VectorMA(self->s.angles, FRAMETIME, self->avelocity, self->s.angles);
|
|
|
|
if(!BoundVelocity(self->velocity) && self->gravity <= 0.0f)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if(self->velocity[2] > 0.0f)
|
|
{
|
|
self->groundentity = NULL;
|
|
}
|
|
else
|
|
{
|
|
if(self->groundentity)
|
|
{ // onground, return without moving
|
|
return;
|
|
}
|
|
}
|
|
|
|
VectorCopy(self->mins, formMove.mins);
|
|
VectorCopy(self->maxs, formMove.maxs);
|
|
|
|
formMove.passEntity = self;
|
|
formMove.clipMask = self->clipmask;
|
|
|
|
MoveEntity_Bounce(self, &formMove);
|
|
|
|
PhysicsCheckWaterTransition(self);
|
|
|
|
gi.linkentity(self);
|
|
|
|
ActivateTriggers(self);
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------
|
|
//Monsters freefall when they don't have a ground entity, otherwise
|
|
//all movement is done with discrete steps.
|
|
|
|
//This is also used for objects that have become still on the ground, but
|
|
//will fall if the floor is pulled out from under them.
|
|
//---------------------------------------------------------------------------------
|
|
static void Physics_StepMove(edict_t *self)
|
|
{
|
|
qboolean hasVel;
|
|
FormMove_t formMove;
|
|
float gravity;
|
|
float friction;
|
|
|
|
assert(self->gravity >= 0.0f);
|
|
|
|
hasVel = Vec3NotZero(self->velocity);
|
|
|
|
// gi.dprintf("hasVel %i\n", hasVel);
|
|
|
|
// gi.dprintf("vel in Physics_StepMove %f, %f, %f\n", self->velocity[0],
|
|
// self->velocity[1], self->velocity[2]);
|
|
|
|
// Apply rotation friction if desired
|
|
if(self->physicsFlags & PF_ROTATIONAL_FRICTION)
|
|
{
|
|
if(!Vec3IsZero(self->avelocity))
|
|
{
|
|
ApplyRotationalFriction(self);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
VectorMA(self->s.angles, FRAMETIME, self->avelocity, self->s.angles);
|
|
}
|
|
|
|
if(self->physicsFlags & PF_RESIZE)
|
|
{
|
|
if(gi.ResizeBoundingForm(self, &formMove))
|
|
{
|
|
self->physicsFlags &= ~PF_RESIZE;
|
|
}
|
|
}
|
|
|
|
gravity = self->gravity * sv_gravity->value;
|
|
|
|
// check for submersion or nograv
|
|
if(self->waterlevel <= 1 && gravity > 0.0f)
|
|
{
|
|
if(self->groundentity)
|
|
{
|
|
friction = self->friction * sv_friction->value;
|
|
|
|
if(!hasVel)
|
|
{
|
|
if(self->groundNormal[2] >= GROUND_NORMAL && self->groundNormal[2] >= (gravity / (friction + gravity)))
|
|
{ // not going anywhere without velocity on ground whose slope this ent won't slide on
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
MoveEntity_Slide(self);
|
|
}
|
|
else
|
|
{
|
|
if(!hasVel)
|
|
{ // not going anywhere without vel
|
|
return;
|
|
}
|
|
|
|
VectorCopy(self->mins, formMove.mins);
|
|
VectorCopy(self->maxs, formMove.maxs);
|
|
|
|
formMove.passEntity = self;
|
|
formMove.clipMask = self->clipmask;
|
|
|
|
MoveEntity_Bounce(self, &formMove);
|
|
}
|
|
|
|
if(!BoundVelocity(self->velocity))
|
|
{
|
|
// gi.dprintf("Doesn't have vel\n");
|
|
|
|
if(hasVel)
|
|
{ // stopped
|
|
QPostMessage(self, G_MSG_RESTSTATE, PRI_PHYSICS, "i", hasVel);
|
|
}
|
|
}
|
|
else if(!hasVel)
|
|
{ // started
|
|
QPostMessage(self, G_MSG_RESTSTATE, PRI_PHYSICS, "i", hasVel);
|
|
}
|
|
|
|
PhysicsCheckWaterTransition(self);
|
|
|
|
gi.linkentity(self);
|
|
|
|
ActivateTriggers(self);
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------
|
|
//---------------------------------------------------------------------------------
|
|
void EntityPhysics(edict_t *self)
|
|
{
|
|
assert(self->inuse);
|
|
|
|
if(self->movetype < 0 || self->movetype >= NUM_PHYSICSTYPES)
|
|
{
|
|
assert(0);
|
|
gi.error ("SV_Physics: bad movetype %i", self->movetype);
|
|
return;
|
|
}
|
|
|
|
physicsFuncs[self->movetype](self);
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------
|
|
// Determines which entity if any the self is on
|
|
// Also zeros z vel and moves onto ground if it was in the air
|
|
//---------------------------------------------------------------------------------
|
|
void CheckEntityOn(edict_t *self)
|
|
{
|
|
vec3_t point;
|
|
FormMove_t formMove;
|
|
|
|
assert(self);
|
|
|
|
if(self->velocity[2] > Z_VEL_NOT_ONGROUND)
|
|
{
|
|
self->groundentity = NULL;
|
|
return;
|
|
}
|
|
|
|
// if the hull point one-quarter unit down is solid the entity is on ground
|
|
point[0] = self->s.origin[0];
|
|
point[1] = self->s.origin[1];
|
|
point[2] = self->s.origin[2] - (PHYSICS_Z_FUDGE + CHECK_BELOW_DIST);
|
|
|
|
VectorCopy(self->mins, formMove.mins);
|
|
VectorCopy(self->maxs, formMove.maxs);
|
|
|
|
formMove.start = self->s.origin;
|
|
formMove.end = point;
|
|
formMove.passEntity = self;
|
|
formMove.clipMask = MASK_MONSTERSOLID;
|
|
|
|
gi.TraceBoundingForm(&formMove);
|
|
|
|
// check steepness
|
|
if(!formMove.trace.ent || (formMove.trace.plane.normal[2] < GROUND_NORMAL && !formMove.trace.startsolid))
|
|
{
|
|
self->groundentity = NULL;
|
|
return;
|
|
}
|
|
|
|
if(!formMove.trace.startsolid && !formMove.trace.allsolid)
|
|
{
|
|
VectorCopy(formMove.trace.endpos, self->s.origin);
|
|
|
|
SetGroundEntFromTrace(self, &formMove.trace);
|
|
|
|
// self->velocity[2] = 0;
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------
|
|
// set move to be based on gravity and velocity, and adjust velocity for gravity
|
|
//---------------------------------------------------------------------------------
|
|
void ApplyGravity(edict_t *self, vec3_t move)
|
|
{
|
|
assert(self);
|
|
|
|
if(move)
|
|
{
|
|
move[2] -= self->gravity * sv_gravity->value * (FRAMETIME * FRAMETIME * 0.5);
|
|
}
|
|
|
|
self->velocity[2] -= self->gravity * sv_gravity->value * FRAMETIME;
|
|
}
|
|
|
|
//Make things that hit each other do some damage
|
|
void DoImpactDamage(edict_t *self, trace_t *trace)
|
|
{
|
|
vec3_t normal, movedir;
|
|
float impact_dmg, total_health, tr_health, self_health, speed;
|
|
int tr_dmg, self_dmg;
|
|
|
|
if(self->impact_debounce_time>level.time)
|
|
return;
|
|
|
|
if(self->svflags&SVF_DO_NO_IMPACT_DMG)
|
|
return;
|
|
|
|
if(self->client)
|
|
{
|
|
if(self->client->playerinfo.edictflags & FL_CHICKEN)
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
|
|
if(self->svflags&SVF_MONSTER||self->mass||self->client)
|
|
{
|
|
speed = VectorLength(self->velocity);
|
|
|
|
if(speed<50)
|
|
return;
|
|
|
|
if(speed < 500 && self->watertype)
|
|
return;
|
|
|
|
impact_dmg = sqrt(speed/10);
|
|
|
|
//monsters dont do impact damage to their own type
|
|
if(self->classID && trace->ent->classID && self->classID == trace->ent->classID)
|
|
return;
|
|
|
|
if(impact_dmg>0)
|
|
{
|
|
VectorSet(normal, 0, 0, 1);
|
|
if(&trace->plane)
|
|
{
|
|
if(trace->plane.normal)
|
|
{
|
|
VectorCopy(trace->plane.normal, normal);
|
|
}
|
|
}
|
|
|
|
VectorSet(movedir, 0, 0, 1);
|
|
if(!Vec3IsZero(self->velocity))
|
|
{
|
|
VectorCopy(self->velocity, movedir);
|
|
VectorNormalize(movedir);
|
|
}
|
|
|
|
if(trace->ent->solid==SOLID_BSP)
|
|
{
|
|
if(self->health>0)
|
|
impact_dmg = impact_dmg * self->health / 100;
|
|
else if(speed<300)
|
|
return;
|
|
|
|
if((!trace->ent->takedamage && self->health*10 > 1000) || self->health<=0)
|
|
tr_health = self->health * 10;
|
|
else
|
|
tr_health = 1000;
|
|
}
|
|
else if(trace->ent->health>0)
|
|
tr_health = trace->ent->health * 0.5;
|
|
else
|
|
tr_health = 1;
|
|
|
|
if(self->health>0)
|
|
self_health = self->health * 2;
|
|
else
|
|
self_health = 2;
|
|
|
|
total_health = self_health + tr_health;
|
|
|
|
if(trace->ent->solid==SOLID_BSP&&!trace->ent->takedamage)
|
|
tr_dmg = 0;
|
|
else
|
|
tr_dmg = floor(impact_dmg * self_health/total_health);
|
|
|
|
self_dmg = floor(impact_dmg - tr_dmg);
|
|
|
|
if(tr_dmg>=1 && trace->ent->takedamage && !(trace->ent->svflags&SVF_TAKE_NO_IMPACT_DMG)&&!(trace->ent->svflags&SVF_BOSS))
|
|
{
|
|
if(skill->value < 2 && self->svflags & SVF_MONSTER && trace->ent->client)
|
|
tr_dmg = ceil(tr_dmg * 0.5);//monsters do a bit less damage to player on normal and easy skill
|
|
if(tr_dmg >= 1)
|
|
{
|
|
T_Damage(trace->ent, self, self, movedir, trace->endpos, normal, tr_dmg, tr_dmg, 0,MOD_CRUSH);
|
|
if(trace->ent->health>0)
|
|
{
|
|
if(trace->ent->client)
|
|
{
|
|
if(tr_dmg > irand(25, 40) - (5 * (skill->value)))
|
|
{
|
|
if(trace->ent->client->playerinfo.lowerseq != ASEQ_KNOCKDOWN)
|
|
P_KnockDownPlayer(&trace->ent->client->playerinfo);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if(self_dmg>=1 && self->takedamage && (speed > 600 || self->health <= 0) && !(self->svflags&SVF_TAKE_NO_IMPACT_DMG) && !(self->svflags&SVF_BOSS) && self->jump_time < level.time)
|
|
{
|
|
if(skill->value && self->client && trace->ent->solid == SOLID_BSP)
|
|
self_dmg = floor(self->dmg * 1.5);//more damage to player from falls
|
|
|
|
if(self_dmg >= 3 && (self->classID != CID_ASSASSIN && self->classID != CID_SSITHRA) || self->health<=0)//but what about ring of repulsion?
|
|
{
|
|
if(self_dmg < 5)
|
|
T_Damage(self, self, self, movedir, trace->endpos, normal, self_dmg, 0, DAMAGE_NO_BLOOD|DAMAGE_AVOID_ARMOR,MOD_FALLING);
|
|
else
|
|
T_Damage(self, self, self, movedir, trace->endpos, normal, self_dmg, 0, DAMAGE_AVOID_ARMOR,MOD_FALLING);
|
|
self->impact_debounce_time = level.time + 0.3;//don't collide again too soon
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------
|
|
// Calls correct collision handling functions for ents involved
|
|
//---------------------------------------------------------------------------------
|
|
void HandleCollision(edict_t *self, trace_t *trace, vec3_t move, int forceful, int flags)
|
|
{
|
|
edict_t *other = trace->ent;
|
|
|
|
// assert(self->solid != SOLID_NOT);
|
|
// assert(other->solid != SOLID_NOT);
|
|
|
|
if(IMPACT_DAMAGE)
|
|
{
|
|
if(flags&CH_ISBLOCKED || flags&CH_BOUNCED)
|
|
{
|
|
DoImpactDamage(self, trace);
|
|
}
|
|
}
|
|
|
|
if(forceful)
|
|
{
|
|
HandleForcefulCollision(self, other, move, forceful);
|
|
}
|
|
|
|
if(flags&CH_ISBLOCKED && self->isBlocked)
|
|
{
|
|
self->isBlocked(self, trace);
|
|
}
|
|
|
|
if(flags&CH_BOUNCED && self->bounced)
|
|
{
|
|
self->bounced(self, trace);
|
|
}
|
|
|
|
if(flags&CH_ISBLOCKING && other->isBlocking)
|
|
{
|
|
trace_t temp = *trace;
|
|
|
|
temp.ent = self;
|
|
VectorInverse(temp.plane.normal);
|
|
|
|
other->isBlocking(other, &temp);
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------
|
|
// forceful < 0 indicates that the forcer shouldn't be knocked back
|
|
//---------------------------------------------------------------------------------------
|
|
void HandleForcefulCollision(edict_t *forcer, edict_t *forcee, vec3_t move, int forceful)
|
|
{
|
|
vec3_t dir, vel;
|
|
float knockback;
|
|
qboolean hitWorld = false;
|
|
|
|
assert(forcee);
|
|
|
|
if(forcee == g_edicts)
|
|
{
|
|
hitWorld = true;
|
|
VectorScale(move, -FRAMES_PER_SECOND*0.8, vel);
|
|
}
|
|
else
|
|
{
|
|
hitWorld = false;
|
|
VectorScale(move, FRAMES_PER_SECOND*0.5, vel);
|
|
}
|
|
|
|
knockback = VectorNormalize2(vel, dir);
|
|
|
|
knockback *= forcer->mass;
|
|
knockback /= KNOCK_BACK_MULTIPLIER;
|
|
|
|
// knock other entity back
|
|
if(!hitWorld)
|
|
{
|
|
PostKnockBack(forcee, dir, knockback, 0);
|
|
Vec3ScaleAssign(-1.0, dir);
|
|
}
|
|
|
|
if(forceful > 0)
|
|
{
|
|
// knock back running ent
|
|
PostKnockBack(forcer, dir, knockback, 0);
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------
|
|
// takes into account gravity, and bounces an ent away from impacts based on elasticity
|
|
// fiction is ignored
|
|
//---------------------------------------------------------------------------------
|
|
void MoveEntity_Bounce(edict_t *self, FormMove_t *formMove)
|
|
{
|
|
vec3_t move, end;
|
|
|
|
VectorScale(self->velocity, FRAMETIME, move);
|
|
|
|
// create the delta move
|
|
if(self->gravity > 0.0f)
|
|
{
|
|
ApplyGravity(self, move);
|
|
}
|
|
|
|
VectorAdd(self->s.origin, move, end);
|
|
|
|
formMove->start = self->s.origin;
|
|
formMove->end = end;
|
|
|
|
gi.TraceBoundingForm(formMove);
|
|
|
|
VectorCopy(formMove->trace.endpos, self->s.origin);
|
|
|
|
// handle bouncing and sliding
|
|
if(formMove->trace.fraction < 1.0)
|
|
{
|
|
BounceVelocity(self->velocity, formMove->trace.plane.normal, self->velocity, self->elasticity);
|
|
|
|
if(self->elasticity > ELASTICITY_SLIDE)
|
|
{
|
|
if((self->velocity[2] < 60.0) && (formMove->trace.plane.normal[2] > GROUND_NORMAL))
|
|
{
|
|
SetGroundEntFromTrace(self, &formMove->trace);
|
|
|
|
VectorClear(self->velocity);
|
|
VectorClear(self->avelocity);
|
|
}
|
|
}
|
|
|
|
HandleCollision(self, &formMove->trace, move, (self->physicsFlags & PF_FORCEFUL_COLLISIONS), CH_STANDARD);
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------
|
|
// Moves an entity sliding or bouncing of off any planes collided with
|
|
//---------------------------------------------------------------------------------
|
|
void MoveEntity_Slide(edict_t *self)
|
|
{
|
|
#define MAX_CLIP_PLANES 5
|
|
#define MAX_BUMPS 4
|
|
#define STEEP_SLOPE_FRICTION_MODIFIER 0.1
|
|
|
|
edict_t *hit;
|
|
int bumpcount;
|
|
vec3_t dir, gdir;
|
|
int numplanes = 0;
|
|
vec3_t primal_velocity, original_velocity, new_velocity;
|
|
int i, j;
|
|
vec3_t end, delta;
|
|
float timeRemaining = FRAMETIME, timeRemaining2, timeMoved;
|
|
static vec3_t planes[MAX_CLIP_PLANES];
|
|
FormMove_t formMove;
|
|
float base_friction, friction, gravity;
|
|
float dist, dot, accel, speed, faccel, gaccel;
|
|
qboolean slide;
|
|
float *groundNormal;
|
|
int fudgeIndex = -1;
|
|
|
|
assert(self->clipmask);
|
|
assert(self);
|
|
|
|
gravity = self->gravity * sv_gravity->value;
|
|
base_friction = self->friction * sv_friction->value;
|
|
// gi.dprintf("Gravity %f, Friction %f, to be applied\n", gravity, friction);
|
|
|
|
// gi.dprintf("Velocity %f, %f, %f\n", self->velocity[0], self->velocity[1], self->velocity[2]);
|
|
// gi.dprintf("speed in %f\n", VectorLength(self->velocity));
|
|
|
|
VectorCopy(self->velocity, original_velocity);
|
|
VectorCopy(self->velocity, primal_velocity);
|
|
|
|
// self->groundentity = NULL;
|
|
|
|
groundNormal = self->groundNormal;
|
|
|
|
if(!self->groundentity)
|
|
{
|
|
groundNormal[2] = 0.0;
|
|
}
|
|
|
|
VectorCopy(self->mins, formMove.mins);
|
|
VectorCopy(self->maxs, formMove.maxs);
|
|
|
|
formMove.start = self->s.origin;
|
|
formMove.passEntity = self;
|
|
formMove.clipMask = self->clipmask;
|
|
|
|
for(bumpcount = 0; bumpcount < MAX_BUMPS; ++bumpcount)
|
|
{
|
|
friction = base_friction;
|
|
|
|
VectorScale(self->velocity, timeRemaining, delta);
|
|
|
|
timeRemaining2 = timeRemaining * timeRemaining;
|
|
|
|
slide = false;
|
|
|
|
if(groundNormal[2])
|
|
{ // on some type of upward facing slope
|
|
assert(groundNormal[2] > 0.0);
|
|
|
|
if(Vec3IsZero(self->velocity))
|
|
{ // no velocity
|
|
// gi.dprintf("no vel on slope\n");
|
|
|
|
if(groundNormal[2] >= GROUND_NORMAL)
|
|
{
|
|
if(groundNormal[2] >= (gravity / (friction + gravity)))
|
|
{ // can't slide
|
|
if(bumpcount)
|
|
{ // not going anywhere
|
|
// gi.dprintf("no vel on no slide slope\n");
|
|
#if 0
|
|
formMove.trace.fraction = 0.0;
|
|
break;
|
|
#else
|
|
return;
|
|
#endif
|
|
}
|
|
else
|
|
{ // check in calling func
|
|
assert(0);
|
|
}
|
|
}
|
|
}
|
|
|
|
// ( |gravity| X groundNormal ) X groundNormal yeilds the vector in the
|
|
//direction of gravity applied to the the slope of groundNormal
|
|
gdir[0] = groundNormal[0]*groundNormal[2];
|
|
gdir[1] = groundNormal[1]*groundNormal[2];
|
|
gdir[2] = -groundNormal[0]*groundNormal[0] - groundNormal[1]*groundNormal[1];
|
|
|
|
VectorNormalize(gdir);
|
|
|
|
dot = DotProduct(gdir, groundNormal);
|
|
|
|
if(dot < -FLOAT_ZERO_EPSILON)
|
|
{ // floating point error, shit, fudge it away from the plane a bit
|
|
// gi.dprintf("Dot %f, Fudge for bump %i and plane %i\n", dot, bumpcount, i);
|
|
fudgeIndex = i;
|
|
VectorMA(self->s.origin, PHYSICS_Z_FUDGE, planes[i], self->s.origin);
|
|
}
|
|
|
|
// dir[2] += 0.0001; // fudge it away from the slope just a little
|
|
|
|
dist = 0;
|
|
|
|
dot = 0;
|
|
|
|
slide = true;
|
|
}
|
|
else
|
|
{
|
|
dist = VectorNormalize2(delta, dir);
|
|
dot = DotProduct(dir, groundNormal);
|
|
|
|
if(dot < -0.05)
|
|
{ // the trace will fail, try to restructure inorder to skip it
|
|
}
|
|
else if(dot < 0.05) // parallel to ground
|
|
{
|
|
slide = true;
|
|
}
|
|
else
|
|
{ // pulling away from ground
|
|
}
|
|
}
|
|
|
|
}
|
|
else
|
|
{ // easy case, fall straight down, no surface friction needed
|
|
}
|
|
|
|
if(slide)
|
|
{ // moving along ground, apply gravity and friction appropriatly
|
|
gdir[0] = groundNormal[0]*groundNormal[2];
|
|
gdir[1] = groundNormal[1]*groundNormal[2];
|
|
gdir[2] = -groundNormal[0]*groundNormal[0] - groundNormal[1]*groundNormal[1];
|
|
|
|
VectorNormalize(gdir);
|
|
|
|
dot = DotProduct(gdir, dir);
|
|
|
|
if(groundNormal[2] < GROUND_NORMAL)
|
|
{ // turn down friction on a steep slope, the theory being that something wouldn't be able to maintain
|
|
// good surface contact on such a slope
|
|
// friction *= STEEP_SLOPE_FRICTION_MODIFIER;
|
|
faccel = -friction * groundNormal[2] * STEEP_SLOPE_FRICTION_MODIFIER;
|
|
}
|
|
else
|
|
{
|
|
faccel = -friction * groundNormal[2];
|
|
}
|
|
|
|
#if 0
|
|
if(accel < 0)
|
|
{
|
|
gi.dprintf("Accel less than zero\n");
|
|
}
|
|
#endif
|
|
|
|
dist += 0.5 * faccel * timeRemaining2;
|
|
|
|
if(dist < 0)
|
|
{
|
|
dist = 0;
|
|
faccel = 0;
|
|
}
|
|
|
|
VectorScale(dir, dist, delta);
|
|
// gi.dprintf("Move slid, accel %f\n", faccel);
|
|
dot = DotProduct(gdir, groundNormal);
|
|
|
|
if(dot < -FLOAT_ZERO_EPSILON)
|
|
{ // floating point error, shit, fudge it away from the plane a bit
|
|
// gi.dprintf("Dot %f, Fudge for bump %i and plane %i\n", dot, bumpcount, i);
|
|
fudgeIndex = i;
|
|
VectorMA(self->s.origin, PHYSICS_Z_FUDGE, planes[i], self->s.origin);
|
|
}
|
|
|
|
if(groundNormal[2] < GROUND_NORMAL)
|
|
{ // turn down friction on a steep slope, the theory being that something wouldn't be able to maintain
|
|
// good surface contact on such a slope
|
|
// friction *= STEEP_SLOPE_FRICTION_MODIFIER;
|
|
gaccel = gravity * (1 - groundNormal[2]) - friction * groundNormal[2] * STEEP_SLOPE_FRICTION_MODIFIER;
|
|
}
|
|
else
|
|
{
|
|
gaccel = gravity * (1 - groundNormal[2]) - friction * groundNormal[2];
|
|
}
|
|
|
|
if(gaccel < 0)
|
|
{
|
|
gaccel = 0;
|
|
}
|
|
|
|
VectorMA(delta, gaccel * 0.5 * timeRemaining2, gdir, delta);
|
|
|
|
if(dist + faccel + gaccel == 0.0)
|
|
{
|
|
VectorClear(self->velocity);
|
|
#if 0
|
|
break;
|
|
#else
|
|
return;
|
|
#endif
|
|
}
|
|
}
|
|
else
|
|
{ // else apply gravity straight down, no friction
|
|
delta[2] -= 0.5 * gravity * timeRemaining2;
|
|
// gi.dprintf("Move dropped, accel\n");
|
|
}
|
|
|
|
VectorCopy(self->s.origin, end);
|
|
Vec3AddAssign(delta, end);
|
|
|
|
formMove.end = end;
|
|
|
|
gi.TraceBoundingForm(&formMove);
|
|
|
|
if(formMove.trace.startsolid)
|
|
{
|
|
if(fudgeIndex != -1)
|
|
{ // undo fudge and let it try that again
|
|
// gi.dprintf("Fudge undone\n");
|
|
VectorMA(self->s.origin, -PHYSICS_Z_FUDGE, planes[fudgeIndex], self->s.origin);
|
|
continue;
|
|
}
|
|
else
|
|
{
|
|
VectorClear(self->velocity);
|
|
|
|
// gi.dprintf("self %i, trace startsolid on bump %i\n", self->s.number, bumpcount);
|
|
return;
|
|
}
|
|
}
|
|
|
|
if(formMove.trace.allsolid)
|
|
{ // entity is trapped in another solid
|
|
VectorClear(self->velocity);
|
|
|
|
self->s.origin[2] += 20;
|
|
|
|
// gi.dprintf("self %d, trace allsolid\n", self->s.number);
|
|
return;
|
|
}
|
|
|
|
timeMoved = timeRemaining * formMove.trace.fraction;
|
|
|
|
if(formMove.trace.fraction > 0)
|
|
{ // actually covered some distance
|
|
VectorCopy(formMove.trace.endpos, self->s.origin);
|
|
|
|
if(slide)
|
|
{
|
|
speed = VectorNormalize2(self->velocity, dir);
|
|
|
|
dot = DotProduct(dir, groundNormal);
|
|
|
|
// assert(Q_fabs(dot) <= 0.05);
|
|
|
|
speed += faccel * timeMoved;
|
|
|
|
#if 0
|
|
if(Q_fabs(speed) < friction * 0.05)
|
|
{
|
|
speed = 0;
|
|
}
|
|
#endif
|
|
|
|
VectorScale(dir, speed, self->velocity);
|
|
// gi.dprintf("Full move, slid, speed %f\n", speed);
|
|
|
|
VectorMA(self->velocity, gaccel * timeMoved, gdir, self->velocity);
|
|
}
|
|
else
|
|
{
|
|
// gi.dprintf("Full move, dropped\n");
|
|
self->velocity[2] -= gravity * timeMoved;
|
|
}
|
|
|
|
if(formMove.trace.fraction == 1)
|
|
{
|
|
break; // moved the entire distance
|
|
}
|
|
|
|
VectorCopy(self->velocity, original_velocity);
|
|
|
|
numplanes = 0;
|
|
fudgeIndex = -1;
|
|
}
|
|
else
|
|
{
|
|
#if 0
|
|
dist = VectorNormalize2(delta, dir);
|
|
|
|
if(Q_fabs(DotProduct(dir, formMove.trace.plane.normal)) < 0.01)
|
|
{
|
|
|
|
}
|
|
#endif
|
|
if(Vec3IsZero(self->velocity) && self->groundNormal[2] >= (gravity / (friction + gravity)))
|
|
{ // no velocity, and the trace failed, not going anywhere on ground the ent can't slide on
|
|
break;
|
|
}
|
|
// gi.dprintf("0 trace with %i bumps and %i planes\n", bumpcount, numplanes);
|
|
}
|
|
|
|
hit = formMove.trace.ent;
|
|
|
|
if(bumpcount == MAX_BUMPS - 1)
|
|
{ // results in isBlocked being called on the last bounced
|
|
HandleCollision(self, &formMove.trace, delta, -1, CH_BOUNCED|CH_STANDARD);
|
|
}
|
|
else
|
|
{
|
|
HandleCollision(self, &formMove.trace, delta, -1, CH_BOUNCED|CH_ISBLOCKING);
|
|
}
|
|
|
|
VectorCopy(formMove.trace.plane.normal, groundNormal);
|
|
|
|
if(groundNormal[2] > 0 && hit->solid == SOLID_BSP) // hit the floor
|
|
{
|
|
self->groundentity = formMove.trace.ent;
|
|
}
|
|
else
|
|
{
|
|
if(groundNormal[2] < 0.0)
|
|
{
|
|
groundNormal[2] = 0.0;
|
|
}
|
|
|
|
self->groundentity = NULL;
|
|
}
|
|
|
|
timeRemaining -= timeMoved;
|
|
|
|
// clipped to another plane
|
|
assert(numplanes < MAX_CLIP_PLANES);
|
|
|
|
if(!numplanes || !VectorCompare(formMove.trace.plane.normal, planes[numplanes-1]))
|
|
{
|
|
VectorCopy(formMove.trace.plane.normal, planes[numplanes]);
|
|
numplanes++;
|
|
}
|
|
else
|
|
{
|
|
// gi.dprintf("Attemping to add identical plane for bump %i\n", bumpcount);
|
|
// gi.dprintf("Identical Plane Fudge for bump %i and plane %i\n", bumpcount, numplanes);
|
|
fudgeIndex = numplanes-1;
|
|
VectorMA(self->s.origin, PHYSICS_Z_FUDGE, planes[fudgeIndex], self->s.origin);
|
|
}
|
|
|
|
// modify original_velocity so it parallels all of the clip planes
|
|
for(i = 0; i < numplanes; ++i)
|
|
{
|
|
float dirMag;
|
|
|
|
assert(Vec3NotZero(planes[i]));
|
|
|
|
#if 0
|
|
#define ACCEL_A // velocity is maintained throughout, otherwise, elasticity is accounted for
|
|
#endif
|
|
|
|
#ifdef ACCEL_A // top version is better for testing sliding with sv_friction at 0
|
|
speed = VectorNormalize2(original_velocity, dir);
|
|
|
|
// gi.dprintf("Speed %f\n", speed);
|
|
|
|
BounceVelocity(dir, planes[i], dir, 1.0001f); // only works with an elasticity a bit greater than 1.0
|
|
// going to need a different func
|
|
// for bouncing stuff
|
|
|
|
dirMag = VectorNormalize(dir);
|
|
|
|
#else
|
|
BounceVelocity(original_velocity, planes[i], new_velocity, self->elasticity);
|
|
|
|
dirMag = speed = VectorNormalize2(new_velocity, dir);
|
|
|
|
#endif // ACCEL_A
|
|
|
|
if(FloatIsZeroEpsilon(dirMag))
|
|
{ // smacked into something exactly head on
|
|
if(planes[i][2] > 0.0)
|
|
{ // slide down slope
|
|
dir[0] = -planes[i][0]*planes[i][2];
|
|
dir[1] = -planes[i][1]*planes[i][2];
|
|
dir[2] = planes[i][0]*planes[i][0] + planes[i][1]*planes[i][1];
|
|
|
|
VectorNormalize(dir);
|
|
}
|
|
else
|
|
{ // drop straight down
|
|
dir[2] = -1.0;
|
|
}
|
|
|
|
}
|
|
|
|
dot = DotProduct(dir, planes[i]);
|
|
|
|
if(dot < -FLOAT_ZERO_EPSILON)
|
|
{ // floating point error, shit, fudge it away from the plane a bit
|
|
// gi.dprintf("Dot %f, Fudge for bump %i and plane %i\n", dot, bumpcount, i);
|
|
fudgeIndex = i;
|
|
VectorMA(self->s.origin, PHYSICS_Z_FUDGE, planes[i], self->s.origin);
|
|
}
|
|
|
|
slide = false;
|
|
|
|
if(planes[i][2] > 0.0)
|
|
{
|
|
if(dot < -0.01)
|
|
{ // the trace will fail, try to restructure inorder to skip it
|
|
#ifdef ACCEL_A
|
|
assert(0); // shouldn't happen
|
|
#endif // ACCEL_A
|
|
}
|
|
else if(Q_fabs(dot) < 0.01) // parallel to surface
|
|
{
|
|
slide = true;
|
|
}
|
|
else
|
|
{ // pulling away from surface
|
|
}
|
|
}
|
|
else
|
|
{ // easy case, fall straight down, no surface friction needed
|
|
}
|
|
|
|
#ifdef ACCEL_A
|
|
if(slide)
|
|
{ // moving along ground, apply gravity and friction appropriatly
|
|
assert(Q_fabs(DotProduct(dir, planes[i])) < 0.1);
|
|
|
|
accel = -friction * planes[i][2] - gravity * dir[2];
|
|
speed += accel * timeRemaining;
|
|
VectorScale(dir, speed, new_velocity);
|
|
// gi.dprintf("Velocity slid, accel %f, speed %f\n", accel, speed);
|
|
}
|
|
else
|
|
{ // else apply gravity straight down, no friction
|
|
VectorScale(dir, speed, new_velocity);
|
|
new_velocity[2] -= gravity * timeRemaining;
|
|
// gi.dprintf("Velocity dropped 1, bump %i, plane %i\n", bumpcount, i);
|
|
// gi.dprintf("dir %f, %f, %f\n", dir[0], dir[1], dir[2]);
|
|
}
|
|
|
|
#endif // ACCEL_A
|
|
|
|
// gi.dprintf("dirMag %f\n", dirMag);
|
|
// gi.dprintf("Speed %f\n", VectorLength(new_velocity));
|
|
|
|
// gi.dprintf("Bounce dot %f\n", dot);
|
|
|
|
for(j = 0; j < numplanes; ++j)
|
|
{
|
|
if(j != i)
|
|
{
|
|
if(DotProduct(new_velocity, planes[j]) <= 0)
|
|
{
|
|
break; // unacceptable slide
|
|
}
|
|
}
|
|
}
|
|
|
|
if(j == numplanes)
|
|
{
|
|
break; // acceptable slide
|
|
}
|
|
}
|
|
|
|
if (i != numplanes)
|
|
{ // good slide
|
|
VectorCopy (new_velocity, self->velocity);
|
|
|
|
// gi.dprintf("Acceptable slide for bump %i\n", bumpcount);
|
|
|
|
}
|
|
else
|
|
{ // go along the crease
|
|
assert(numplanes);
|
|
|
|
if (numplanes != 2)
|
|
{
|
|
// gi.dprintf ("slide, numplanes == %i\n",numplanes);
|
|
VectorClear(self->velocity);
|
|
return;
|
|
}
|
|
|
|
// gi.dprintf("Unacceptable slide for bump %i\n", bumpcount);
|
|
|
|
CrossProduct (planes[0], planes[1], dir);
|
|
|
|
speed = VectorLength(self->velocity);
|
|
|
|
if(dir[2] <= 0)
|
|
{
|
|
#ifdef ACCEL_A
|
|
accel = -friction * (1 - dir[2]) - gravity * dir[2];
|
|
speed += accel * timeRemaining;
|
|
#else
|
|
slide = true;
|
|
#endif // ACCEL_A
|
|
}
|
|
|
|
VectorScale(dir, speed, self->velocity);
|
|
}
|
|
|
|
#ifndef ACCEL_A
|
|
speed = VectorNormalize2(self->velocity, dir);
|
|
|
|
if(slide)
|
|
{ // moving along ground, apply gravity and friction appropriatly
|
|
accel = -friction * (1 - dir[2]) - gravity * dir[2];
|
|
speed += accel * timeRemaining;
|
|
VectorScale(dir, speed, new_velocity);
|
|
// gi.dprintf("Velocity slid, accel %f, speed %f\n", accel, speed);
|
|
}
|
|
else
|
|
{ // else apply gravity straight down, no friction
|
|
VectorScale(dir, speed, new_velocity);
|
|
new_velocity[2] -= gravity * timeRemaining;
|
|
// gi.dprintf("Velocity dropped 2, bump %i, plane %i\n", bumpcount, i);
|
|
// gi.dprintf("dir %f, %f, %f\n", dir[0], dir[1], dir[2]);
|
|
}
|
|
|
|
#endif // ACCEL_A
|
|
|
|
// if velocity is against the original velocity, stop dead
|
|
// to avoid tiny occilations in sloping corners
|
|
#if 0 // haven't seen a problem with it yet. . .
|
|
if(DotProduct(self->velocity, primal_velocity) <= 0)
|
|
{
|
|
gi.dprintf("Velocity was cleared at bump %i\n", bumpcount);
|
|
|
|
VectorClear(self->velocity);
|
|
break;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
if(formMove.trace.fraction < 1)
|
|
{
|
|
HandleCollision(self, &formMove.trace, delta, -1, CH_STANDARD);
|
|
|
|
if(Vec3NotZero(self->velocity))
|
|
{
|
|
VectorClear(self->velocity);
|
|
// gi.dprintf("Unsuccesful move with %i bumps and %i planes\n", bumpcount, numplanes);
|
|
}
|
|
|
|
if(formMove.trace.plane.normal[2] > GROUND_NORMAL) // hit the floor
|
|
{
|
|
SetGroundEntFromTrace(self, &formMove.trace);
|
|
return;
|
|
}
|
|
}
|
|
CheckEntityOn(self);
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------
|
|
// Searches for any triggers in the entities bounding box and their touch function
|
|
//---------------------------------------------------------------------------------
|
|
void ActivateTriggers(edict_t *self)
|
|
{
|
|
int num;
|
|
edict_t *hit;
|
|
GenericUnion4_t found;
|
|
SinglyLinkedList_t list;
|
|
|
|
// dead things don't activate triggers
|
|
if(self->deadState != DEAD_NO)
|
|
{
|
|
return;
|
|
}
|
|
|
|
SLList_DefaultCon(&list); // this should be global, initialized at startup
|
|
|
|
num = gi.FindEntitiesInBounds(self->mins, self->maxs, &list, AREA_TRIGGERS);
|
|
|
|
while(!SLList_IsEmpty(&list))
|
|
{
|
|
found = SLList_Pop(&list);
|
|
|
|
hit = found.t_edict_p;
|
|
|
|
if(!hit->inuse || !hit->touch)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
hit->touch(hit, self, NULL, NULL);
|
|
}
|
|
|
|
SLList_Des(&list); // kill on shut down
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------
|
|
//---------------------------------------------------------------------------------
|
|
void ApplyRotationalFriction(edict_t *self)
|
|
{
|
|
int i;
|
|
float adjustment;
|
|
|
|
VectorMA(self->s.angles, FRAMETIME, self->avelocity, self->s.angles);
|
|
|
|
adjustment = FRAMETIME * FRICTION_STOPSPEED * FRICTION_SURFACE;
|
|
|
|
for(i = 0; i < 3; ++i)
|
|
{
|
|
if(self->avelocity[i] > 0.0)
|
|
{
|
|
self->avelocity[i] -= adjustment;
|
|
|
|
if(self->avelocity[i] < 0.0)
|
|
{
|
|
self->avelocity[i] = 0.0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
self->avelocity[i] += adjustment;
|
|
|
|
if(self->avelocity[i] > 0.0)
|
|
{
|
|
self->avelocity[i] = 0.0;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------
|
|
// Sets the groundentity info contained in self based on trace
|
|
//---------------------------------------------------------------------------------
|
|
void SetGroundEntFromTrace(edict_t *self, trace_t *trace)
|
|
{
|
|
assert(self);
|
|
|
|
assert(trace->plane.normal[2] > GROUND_NORMAL);
|
|
|
|
self->groundentity = trace->ent;
|
|
|
|
self->groundentity_linkcount = trace->ent->linkcount;
|
|
|
|
VectorCopy(trace->plane.normal, self->groundNormal);
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------
|
|
// Sets the blockingEntity info contained in self based on trace
|
|
//---------------------------------------------------------------------------------
|
|
void SetBlockingEntFromTrace(edict_t *self, trace_t *trace)
|
|
{
|
|
assert(self);
|
|
|
|
self->blockingEntity = trace->ent;
|
|
|
|
self->blockingEntity_linkcount = trace->ent->linkcount;
|
|
|
|
VectorCopy(trace->plane.normal, self->blockingNormal);
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------
|
|
// Determines if the self is in the water; if so, how submereged the self is
|
|
//---------------------------------------------------------------------------------
|
|
void CheckInWater(FormMove_t *formMove)
|
|
{
|
|
int contents;
|
|
float halfHeight;
|
|
|
|
assert(formMove);
|
|
|
|
contents = gi.GetContentsAtPoint(formMove->start);
|
|
|
|
if(!(contents & MASK_WATER))
|
|
{
|
|
formMove->waterLevel = 0;
|
|
formMove->waterType = 0;
|
|
return;
|
|
}
|
|
|
|
formMove->waterType = contents;
|
|
formMove->waterLevel = 1;
|
|
|
|
halfHeight = (formMove->maxs[2] - (formMove->mins[2] + 1.0)) * 0.5;
|
|
|
|
formMove->start[2] += halfHeight;
|
|
|
|
contents = gi.GetContentsAtPoint(formMove->start);
|
|
|
|
if(!(contents & MASK_WATER))
|
|
{
|
|
return;
|
|
}
|
|
|
|
formMove->waterLevel = 2;
|
|
|
|
formMove->start[2] += halfHeight;
|
|
|
|
contents = gi.GetContentsAtPoint(formMove->start);
|
|
|
|
if(contents & MASK_WATER)
|
|
{
|
|
formMove->waterLevel = 3;
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------
|
|
// Returns false if any part of the bottom of the entity is off an edge that
|
|
// is not a staircase.
|
|
//---------------------------------------------------------------------------------
|
|
qboolean CheckFooting(edict_t *self, vec3_t origin)
|
|
{
|
|
vec3_t mins, maxs, start, stop;
|
|
int x, y;
|
|
float mid, bottom;
|
|
qboolean solid;
|
|
float stepHeight = classStatics[self->classID].moveInfo->stepHeight;
|
|
FormMove_t formMove;
|
|
|
|
VectorAdd(origin, self->mins, mins);
|
|
VectorAdd(origin, self->maxs, maxs);
|
|
|
|
// if all of the points under the corners are solid world, don't bother
|
|
// with the tougher checks
|
|
start[2] = mins[2] - 1;
|
|
|
|
for(x = 0; x <= 1; ++x)
|
|
{
|
|
for(y = 0; y <= 1; ++y)
|
|
{
|
|
start[0] = x ? maxs[0] : mins[0];
|
|
start[1] = y ? maxs[1] : mins[1];
|
|
|
|
if(gi.GetContentsAtPoint(start) != CONTENTS_SOLID)
|
|
{
|
|
solid = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
if(solid)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
start[0] = stop[0] = (mins[0] + maxs[0])*0.5;
|
|
start[1] = stop[1] = (mins[1] + maxs[1])*0.5;
|
|
start[2] = mins[2];
|
|
|
|
stop[2] = start[2] - 2*stepHeight;
|
|
|
|
VectorClear(formMove.mins);
|
|
VectorClear(formMove.maxs);
|
|
|
|
formMove.start = start;
|
|
formMove.end = stop;
|
|
formMove.passEntity = self;
|
|
formMove.clipMask = MASK_MONSTERSOLID;
|
|
|
|
gi.TraceBoundingForm(&formMove);
|
|
|
|
if(formMove.trace.fraction == 1.0)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
mid = bottom = formMove.trace.endpos[2];
|
|
|
|
for(x = 0; x <= 1; ++x)
|
|
{
|
|
for(y = 0; y <= 1; ++y)
|
|
{
|
|
start[0] = stop[0] = x ? maxs[0] : mins[0];
|
|
start[1] = stop[1] = y ? maxs[1] : mins[1];
|
|
|
|
formMove.start = start;
|
|
formMove.end = stop;
|
|
|
|
gi.TraceBoundingForm(&formMove);
|
|
|
|
if(formMove.trace.fraction != 1.0 && formMove.trace.endpos[2] > bottom)
|
|
{
|
|
bottom = formMove.trace.endpos[2];
|
|
}
|
|
|
|
if(formMove.trace.fraction == 1.0 || mid - formMove.trace.endpos[2] > stepHeight)
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static void DiscreteMove_Step_FailedDueToCollision(edict_t *self, vec3_t move, FormMove_t *formMove, qboolean forceful)
|
|
{
|
|
if(!(formMove->processFlags & PPF_INFO_GRAB))
|
|
{
|
|
SetBlockingEntFromTrace(self, &formMove->trace);
|
|
HandleCollision(self, &formMove->trace, move, (self->physicsFlags & PF_FORCEFUL_COLLISIONS) & forceful, CH_STANDARD);
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------
|
|
// Attempts to move self as specified by move. Allows walking on slopes up to
|
|
// GROUND_NORMAL, steps up or down to classStatics[self->classID].self->stepHeight,
|
|
// will drio off edges up to classStatics[self->classID].self->dropHeight,
|
|
// won't enter water higher than level 2
|
|
//
|
|
// Supports PPF_INFO_GRAB for checking a step without actually moving
|
|
//---------------------------------------------------------------------------------
|
|
qboolean DiscreteMove_Step(edict_t *self, vec3_t move, FormMove_t *formMove)
|
|
{
|
|
vec3_t neworg, start, end;
|
|
float traceLength;
|
|
vec3_t test;
|
|
qboolean stepUp = false, setGroundEnt = false;
|
|
|
|
assert(self);
|
|
assert(formMove);
|
|
|
|
if(self->physicsFlags & PF_RESIZE)
|
|
{
|
|
if(gi.ResizeBoundingForm(self, formMove))
|
|
{
|
|
self->physicsFlags &= ~PF_RESIZE;
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
VectorAdd(self->s.origin, move, neworg);
|
|
|
|
#ifdef NONCLIENT_PRIMITIVE_DISPLAY_HACK
|
|
{
|
|
paletteRGBA_t color;
|
|
vec3_t origin;
|
|
|
|
color.r = 255;
|
|
color.g = 0;
|
|
color.b = 0;
|
|
color.a = 255;
|
|
|
|
VectorCopy(self->s.origin, origin);
|
|
origin[2] += 100.0;
|
|
AddServerParticle(origin, color, 2, 5);
|
|
|
|
VectorCopy(neworg, origin);
|
|
origin[2] += 100.0;
|
|
AddServerParticle(origin, color, 2, 5);
|
|
}
|
|
#endif
|
|
|
|
start[0] = end[0] = neworg[0];
|
|
start[1] = end[1] = neworg[1];
|
|
start[2] = neworg[2] + formMove->stepHeight;
|
|
|
|
if(neworg[2] > self->s.origin[2])
|
|
{
|
|
end[2] = self->s.origin[2] - formMove->dropHeight;
|
|
stepUp = true;
|
|
}
|
|
else
|
|
{
|
|
end[2] = neworg[2] - formMove->dropHeight;
|
|
}
|
|
|
|
traceLength = start[2] - end[2];
|
|
|
|
VectorCopy(self->mins, formMove->mins);
|
|
VectorCopy(self->maxs, formMove->maxs);
|
|
|
|
formMove->start = start;
|
|
formMove->end = end;
|
|
formMove->passEntity = self;
|
|
formMove->clipMask = self->clipmask;
|
|
|
|
gi.TraceBoundingForm(formMove);
|
|
|
|
if(formMove->trace.allsolid)
|
|
{
|
|
end[2] = neworg[2] + 0.25f;
|
|
|
|
// find out what's blocking it
|
|
formMove->start = self->s.origin;
|
|
|
|
gi.TraceBoundingForm(formMove);
|
|
|
|
DiscreteMove_Step_FailedDueToCollision(self, move, formMove, true);
|
|
return false;
|
|
}
|
|
|
|
if(formMove->trace.startsolid)
|
|
{
|
|
start[2] = neworg[2] + 0.25f;
|
|
|
|
traceLength = start[2] - end[2];
|
|
|
|
formMove->start = start;
|
|
|
|
gi.TraceBoundingForm(formMove);
|
|
|
|
if(formMove->trace.allsolid)
|
|
{
|
|
// find out what's blocking it
|
|
formMove->start = self->s.origin;
|
|
formMove->end = neworg;
|
|
|
|
gi.TraceBoundingForm(formMove);
|
|
|
|
DiscreteMove_Step_FailedDueToCollision(self, move, formMove, true);
|
|
return false;
|
|
}
|
|
|
|
if(formMove->trace.startsolid)
|
|
{
|
|
// find out what's blocking it
|
|
formMove->start = self->s.origin;
|
|
formMove->end = neworg;
|
|
|
|
gi.TraceBoundingForm(formMove);
|
|
|
|
DiscreteMove_Step_FailedDueToCollision(self, move, formMove, true);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
test[0] = formMove->trace.endpos[0];
|
|
test[1] = formMove->trace.endpos[1];
|
|
test[2] = formMove->trace.endpos[2] + self->mins[2] + 1;
|
|
|
|
formMove->start = test;
|
|
|
|
CheckInWater(formMove);
|
|
|
|
// don't go into deep water
|
|
if(formMove->waterLevel > 2)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
// don't walk off edges
|
|
if(formMove->trace.fraction == 1)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
#if 0
|
|
// check point traces down for dangling corners
|
|
if(!CheckFooting(self, formMove->trace.endpos))
|
|
{
|
|
return false;
|
|
}
|
|
#endif
|
|
|
|
if(stepUp || formMove->trace.endpos[2] >= neworg[2] - formMove->dropHeight)
|
|
{
|
|
// make sure its still on the ground
|
|
if(formMove->trace.plane.normal[2] > GROUND_NORMAL)
|
|
{
|
|
setGroundEnt = true;
|
|
}
|
|
else
|
|
{
|
|
DiscreteMove_Step_FailedDueToCollision(self, move, formMove, !stepUp);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if(!(formMove->processFlags & PPF_INFO_GRAB))
|
|
{ // finalize new position
|
|
self->waterlevel = formMove->waterLevel;
|
|
self->watertype = formMove->waterType;
|
|
|
|
self->flags &= ~FL_INWATER;
|
|
|
|
if(formMove->waterLevel)
|
|
{
|
|
self->flags |= FL_INWATER;
|
|
}
|
|
|
|
if(setGroundEnt)
|
|
{ // step up or down maintaing contact with the ground
|
|
SetGroundEntFromTrace(self, &formMove->trace);
|
|
VectorCopy(formMove->trace.endpos, self->s.origin);
|
|
}
|
|
else
|
|
{ // drop off and edge
|
|
self->groundentity = NULL;
|
|
VectorCopy(neworg, self->s.origin);
|
|
|
|
// should probably set a velocity here also
|
|
}
|
|
|
|
gi.linkentity(self);
|
|
ActivateTriggers(self);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void CreateMove_Step(edict_t *self, vec3_t move, float dist)
|
|
{
|
|
float yaw, pitch;
|
|
float cosYaw;
|
|
float sinYaw;
|
|
|
|
assert(self->groundentity); // need a groundentity if groundNormal is to be used
|
|
|
|
// assert(self->groundNormal[2] > GROUND_NORMAL);
|
|
|
|
yaw = self->s.angles[YAW]*ANGLE_TO_RAD + self->yawOffset;
|
|
|
|
cosYaw = cos(yaw);
|
|
sinYaw = sin(yaw);
|
|
|
|
move[0] = cosYaw;
|
|
move[1] = sinYaw;
|
|
move[2] = 0;
|
|
|
|
pitch = acos(DotProduct(self->groundNormal, move)) - ANGLE_90;
|
|
|
|
// adjust for sloped ground
|
|
if(pitch)
|
|
{
|
|
float cosPitch;
|
|
|
|
cosPitch = cos(pitch);
|
|
|
|
move[0] = cosYaw*cosPitch;
|
|
move[1] = sinYaw*cosPitch;
|
|
move[2] = sin(pitch);
|
|
}
|
|
|
|
Vec3ScaleAssign(dist, move);
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------
|
|
//The move will be adjusted for slopes and stairs, but if the move isn't
|
|
//possible, no move is done, false is returned
|
|
//
|
|
// FINISH ME!!! These need a lot more work.
|
|
//
|
|
//---------------------------------------------------------------------------------
|
|
|
|
void AttemptAdjustOriginToValidPosition(vec3_t adjustFrom, edict_t *self, vec3_t origin)
|
|
{
|
|
FormMove_t formMove;
|
|
|
|
VectorCopy(self->mins, formMove.mins);
|
|
VectorCopy(self->maxs, formMove.maxs);
|
|
|
|
formMove.start = adjustFrom;
|
|
formMove.end = origin;
|
|
formMove.passEntity = self;
|
|
formMove.clipMask = self->clipmask;
|
|
|
|
gi.TraceBoundingForm(&formMove);
|
|
|
|
if(!(formMove.trace.allsolid || formMove.trace.startsolid))
|
|
{
|
|
VectorCopy(formMove.trace.endpos, origin);
|
|
}
|
|
else
|
|
{
|
|
VectorCopy(adjustFrom, origin);
|
|
}
|
|
}
|
|
|
|
qboolean CheckAnimMove(edict_t *self, vec3_t origin, vec3_t move, float *dist, float attemptDist, int recursionLevel)
|
|
{
|
|
vec3_t desiredOrigin, end, temp;
|
|
float stepHeight, tempDist;
|
|
FormMove_t formMove, stepFormMove;
|
|
vec3_t test;
|
|
vec3_t adjustFrom, fudgeOrigin;
|
|
|
|
if(!checkanim->value)
|
|
{ // globally disablled, allow full move
|
|
*dist = attemptDist;
|
|
return true;
|
|
}
|
|
|
|
#if 1
|
|
if(recursionLevel > 10)
|
|
{
|
|
// gi.dprintf("trace startsolid in CheckAnimMove, recursion > 10, failing\n");
|
|
return false;
|
|
}
|
|
#endif
|
|
|
|
stepHeight = classStatics[self->classID].moveInfo->stepHeight;
|
|
|
|
VectorCopy(origin, fudgeOrigin);
|
|
fudgeOrigin[2] += 1.0;
|
|
|
|
VectorAdd(fudgeOrigin, move, desiredOrigin);
|
|
|
|
VectorCopy(self->mins, formMove.mins);
|
|
VectorCopy(self->maxs, formMove.maxs);
|
|
|
|
formMove.start = origin;
|
|
formMove.end = desiredOrigin;
|
|
formMove.passEntity = self;
|
|
formMove.clipMask = self->clipmask;
|
|
|
|
gi.TraceBoundingForm(&formMove);
|
|
|
|
if(formMove.trace.startsolid)
|
|
{
|
|
if(!Vec3IsZero(formMove.trace.plane.normal))
|
|
{
|
|
if(formMove.trace.plane.normal[2] > 0)
|
|
{
|
|
// gi.dprintf("trace startsolid in CheckAnimMove, attempting to adjust with normal\n");
|
|
|
|
VectorMA(origin, 20.0, formMove.trace.plane.normal, adjustFrom);
|
|
|
|
AttemptAdjustOriginToValidPosition(adjustFrom, self, origin);
|
|
|
|
return CheckAnimMove(self, origin, move, dist, attemptDist, recursionLevel + 1);
|
|
}
|
|
else
|
|
{
|
|
// gi.dprintf("trace startsolid in CheckAnimMove, bad normal\n");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// gi.dprintf("trace startsolid in CheckAnimMove, attempting to adjust without normal\n");
|
|
|
|
VectorCopy(origin, adjustFrom);
|
|
|
|
adjustFrom[2] += 20.0;
|
|
|
|
AttemptAdjustOriginToValidPosition(adjustFrom, self, origin);
|
|
|
|
return CheckAnimMove(self, origin, move, dist, attemptDist, recursionLevel + 1);
|
|
}
|
|
}
|
|
|
|
if(formMove.trace.allsolid)
|
|
{
|
|
// gi.dprintf("trace allsolid in CheckAnimMove, attempting to adjust without normal\n");
|
|
|
|
VectorCopy(origin, adjustFrom);
|
|
|
|
adjustFrom[2] += 20.0;
|
|
|
|
AttemptAdjustOriginToValidPosition(adjustFrom, self, origin);
|
|
|
|
return CheckAnimMove(self, origin, move, dist, attemptDist, recursionLevel + 1);
|
|
}
|
|
|
|
VectorScale(move, formMove.trace.fraction, temp);
|
|
|
|
tempDist = VectorLength(temp);
|
|
|
|
// gi.dprintf("tempDist %f\n", tempDist);
|
|
|
|
if(tempDist + *dist < TRACE_FRACTION_FAIL_FOR_WHOLE_MOVE_CHECK*attemptDist)
|
|
{
|
|
if(recursionLevel)
|
|
{
|
|
// don't try to step on recursive calls if the surface isn't ground, or
|
|
// it was only able to move a little forward
|
|
if(formMove.trace.plane.normal[2] <= GROUND_NORMAL ||
|
|
tempDist < stepHeight)
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
*dist += tempDist;
|
|
|
|
// gi.dprintf("Recursion with fraction %f dist %f\n", formMove.trace.fraction, *dist);
|
|
|
|
VectorAdd(origin, temp, adjustFrom);
|
|
|
|
adjustFrom[2] += stepHeight;
|
|
|
|
Vec3ScaleAssign(1.0 - formMove.trace.fraction, move);
|
|
|
|
return CheckAnimMove(self, adjustFrom, move, dist, attemptDist, recursionLevel + 1);
|
|
}
|
|
|
|
VectorMA(fudgeOrigin, formMove.trace.fraction, move, desiredOrigin);
|
|
|
|
// Check for going off an edge
|
|
end[0] = desiredOrigin[0];
|
|
end[1] = desiredOrigin[1];
|
|
|
|
if(desiredOrigin[2] > origin[2])
|
|
{
|
|
end[2] = origin[2] - stepHeight;
|
|
}
|
|
else
|
|
{
|
|
end[2] = desiredOrigin[2] - stepHeight;
|
|
}
|
|
|
|
VectorCopy(formMove.mins, stepFormMove.mins);
|
|
VectorCopy(formMove.maxs, stepFormMove.maxs);
|
|
|
|
stepFormMove.start = desiredOrigin;
|
|
stepFormMove.end = end;
|
|
stepFormMove.passEntity = self;
|
|
stepFormMove.clipMask = self->clipmask;
|
|
|
|
gi.TraceBoundingForm(&stepFormMove);
|
|
|
|
if(stepFormMove.trace.startsolid)
|
|
{
|
|
// gi.dprintf("stepTrace startsolid CheckAnimMove\n");
|
|
return false;
|
|
// stepFormMove.trace.fraction = 0.0;
|
|
}
|
|
|
|
if(stepFormMove.trace.allsolid)
|
|
{
|
|
// gi.dprintf("stepTrace allsolid CheckAnimMove\n");
|
|
return false;
|
|
// stepFormMove.trace.fraction = 0.0;
|
|
}
|
|
|
|
// may have walked off an edge
|
|
if(stepFormMove.trace.fraction == 1.0)
|
|
{
|
|
VectorCopy(origin, adjustFrom);
|
|
|
|
adjustFrom[2] = stepFormMove.trace.endpos[2];
|
|
|
|
stepFormMove.start = stepFormMove.trace.endpos;
|
|
stepFormMove.end = adjustFrom;
|
|
|
|
gi.TraceBoundingForm(&stepFormMove);
|
|
|
|
VectorCopy(temp, move);
|
|
|
|
Vec3ScaleAssign(1.0 - stepFormMove.trace.fraction, temp);
|
|
|
|
tempDist = VectorLength(temp);
|
|
|
|
if(recursionLevel)
|
|
{
|
|
// don't try to step on recursive calls if the surface isn't ground, or
|
|
// it was only able to move a little forward
|
|
if(stepFormMove.trace.plane.normal[2] <= GROUND_NORMAL ||
|
|
tempDist < stepHeight)
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
*dist += tempDist;
|
|
|
|
Vec3AddAssign(temp, adjustFrom);
|
|
|
|
// gi.dprintf("Step off recursion with fraction %f dist %f\n", stepFormMove.trace.fraction, *dist);
|
|
|
|
Vec3ScaleAssign(formMove.trace.fraction, move);
|
|
|
|
return CheckAnimMove(self, adjustFrom, move, dist, attemptDist, recursionLevel + 1);
|
|
}
|
|
|
|
// stepped up onto something it can't stand on
|
|
if(stepFormMove.trace.plane.normal[2] <= GROUND_NORMAL)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
// don't go into water
|
|
if(self->waterlevel == 0.0)
|
|
{
|
|
test[0] = stepFormMove.trace.endpos[0];
|
|
test[1] = stepFormMove.trace.endpos[1];
|
|
test[2] = stepFormMove.trace.endpos[2] + self->mins[2] + 1;
|
|
|
|
stepFormMove.start = test;
|
|
|
|
CheckInWater(&stepFormMove);
|
|
|
|
// probably want to actually let monsters wade
|
|
if(self->flags & FL_INWATER)
|
|
{
|
|
self->waterlevel = 0;
|
|
self->watertype = 0;
|
|
|
|
self->flags &= ~FL_INWATER;
|
|
return false;
|
|
}
|
|
}
|
|
|
|
VectorScale(move, formMove.trace.fraction, temp);
|
|
|
|
*dist += VectorLength(temp);
|
|
|
|
return true;
|
|
}
|
|
|
|
edict_t *TestEntityPosition(edict_t *self)
|
|
{
|
|
FormMove_t formMove;
|
|
|
|
VectorCopy(self->mins, formMove.mins);
|
|
VectorCopy(self->maxs, formMove.maxs);
|
|
|
|
formMove.start = self->s.origin;
|
|
formMove.end = self->s.origin;
|
|
formMove.passEntity = self;
|
|
formMove.clipMask = self->clipmask;
|
|
|
|
gi.TraceBoundingForm(&formMove);
|
|
|
|
if(formMove.trace.startsolid)
|
|
{
|
|
return g_edicts;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
typedef struct
|
|
{
|
|
edict_t *ent;
|
|
vec3_t origin;
|
|
vec3_t angles;
|
|
float deltayaw;
|
|
} pushed_t;
|
|
pushed_t pushed[MAX_EDICTS], *pushed_p;
|
|
|
|
edict_t *obstacle;
|
|
|
|
#define MAX_MATRIX (3)
|
|
|
|
static void BackSub(float a[][MAX_MATRIX],int *indx,float *b,int sz)
|
|
{
|
|
int i,j,ii=-1,ip;
|
|
float sum;
|
|
for (i=0;i<sz;i++)
|
|
{
|
|
ip=indx[i];
|
|
sum=b[ip];
|
|
b[ip]=b[i];
|
|
if (ii>=0)
|
|
{
|
|
for (j=ii;j<i;j++)
|
|
sum-=a[i][j]*b[j];
|
|
}
|
|
else if (sum!=0.0)
|
|
{
|
|
ii=i;
|
|
}
|
|
b[i]=sum;
|
|
}
|
|
for (i=sz-1;i>=0;i--)
|
|
{
|
|
sum=b[i];
|
|
for (j=i+1;j<sz;j++)
|
|
sum-=a[i][j]*b[j];
|
|
b[i]=sum/a[i][i];
|
|
}
|
|
}
|
|
|
|
#define EPS_MATRIX (1E-15)
|
|
|
|
static int LUDecomposition(float a[][MAX_MATRIX],int indx[],int sz,float *d)
|
|
{
|
|
int i,imax,j,k;
|
|
float big,dum,sum,temp;
|
|
float vv[MAX_MATRIX];
|
|
*d=1;
|
|
for (i=0;i<sz;i++)
|
|
{
|
|
big=0.;
|
|
for (j=0;j<sz;j++)
|
|
if ((temp=fabs(a[i][j]))>big)
|
|
big=temp;
|
|
if (big<EPS_MATRIX)
|
|
return 0;
|
|
vv[i]=1.0/big;
|
|
}
|
|
imax = 0;//should have a def value just to make sure
|
|
for (j=0;j<sz;j++)
|
|
{
|
|
for (i=0;i<j;i++)
|
|
{
|
|
sum=a[i][j];
|
|
for (k=0;k<i;k++)
|
|
sum-=a[i][k]*a[k][j];
|
|
a[i][j]=sum;
|
|
}
|
|
big=0.;
|
|
for (i=j;i<sz;i++)
|
|
{
|
|
sum=a[i][j];
|
|
for (k=0;k<j;k++)
|
|
sum-=a[i][k]*a[k][j];
|
|
a[i][j]=sum;
|
|
if ((dum=vv[i]*fabs(sum))>=big)
|
|
{
|
|
big=dum;
|
|
imax=i;
|
|
}
|
|
}
|
|
if (j!=imax)
|
|
{
|
|
for (k=0;k<sz;k++)
|
|
{
|
|
dum=a[imax][k];
|
|
a[imax][k]=a[j][k];
|
|
a[j][k]=dum;
|
|
}
|
|
vv[imax]=vv[j];
|
|
*d=-(*d);
|
|
}
|
|
indx[j]=imax;
|
|
if (fabs(a[j][j])<EPS_MATRIX)
|
|
return 0;
|
|
if (j!=sz-1)
|
|
{
|
|
dum=1.0/a[j][j];
|
|
for (i=j+1;i<sz;i++)
|
|
a[i][j]*=dum;
|
|
}
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
============
|
|
PushEntities
|
|
|
|
Objects need to be moved back on a failed push,
|
|
otherwise riders would continue to slide.
|
|
============
|
|
*/
|
|
|
|
#define GIL_PUSH 1
|
|
qboolean PushEntities(edict_t *pusher, vec3_t move, vec3_t amove)
|
|
{
|
|
int e;
|
|
edict_t *check, *block;
|
|
vec3_t mins, maxs;
|
|
pushed_t *p;
|
|
vec3_t org, org2, move2, forward, right, up,holdOrg;
|
|
|
|
#if GIL_PUSH
|
|
float a[3][3];
|
|
vec3_t forwardInv, rightInv, upInv,OrgOrg;
|
|
int indx[3];
|
|
float d;
|
|
#endif
|
|
// Commented out for Mr Rick
|
|
#if 0
|
|
// clamp the move to 1/8 units, so the position will
|
|
// be accurate for client side prediction
|
|
Vec3ScaleAssign(8.0, move);
|
|
VectorRound(move);
|
|
Vec3ScaleAssign(0.125, move);
|
|
#endif
|
|
|
|
// find the bounding box
|
|
|
|
if (move[0]<0)
|
|
{
|
|
mins[0]=pusher->absmin[0]+move[0];
|
|
maxs[0]=pusher->absmax[0];
|
|
}
|
|
else
|
|
{
|
|
mins[0]=pusher->absmin[0];
|
|
maxs[0]=pusher->absmax[0]+move[0];
|
|
}
|
|
if (move[1]<0)
|
|
{
|
|
mins[1]=pusher->absmin[1]+move[1];
|
|
maxs[1]=pusher->absmax[1];
|
|
}
|
|
else
|
|
{
|
|
mins[1]=pusher->absmin[1];
|
|
maxs[1]=pusher->absmax[1]+move[1];
|
|
}
|
|
if (move[2]<0)
|
|
{
|
|
mins[2]=pusher->absmin[2]+move[2];
|
|
maxs[2]=pusher->absmax[2];
|
|
}
|
|
else
|
|
{
|
|
mins[2]=pusher->absmin[2];
|
|
maxs[2]=pusher->absmax[2]+move[2];
|
|
}
|
|
|
|
// we need this for pushing things later
|
|
#if GIL_PUSH
|
|
VectorCopy(pusher->s.angles, org);
|
|
Vec3AddAssign(amove, org);
|
|
AngleVectors(org, forward, right, up);
|
|
/*
|
|
a[0][0]=forward[0];
|
|
a[0][1]=forward[1];
|
|
a[0][2]=forward[2];
|
|
a[1][0]=-right[0];
|
|
a[1][1]=-right[1];
|
|
a[1][2]=-right[2];
|
|
a[2][0]=up[0];
|
|
a[2][1]=up[1];
|
|
a[2][2]=up[2];
|
|
*/
|
|
a[0][0]=forward[0];
|
|
a[1][0]=forward[1];
|
|
a[2][0]=forward[2];
|
|
a[0][1]=-right[0];
|
|
a[1][1]=-right[1];
|
|
a[2][1]=-right[2];
|
|
a[0][2]=up[0];
|
|
a[1][2]=up[1];
|
|
a[2][2]=up[2];
|
|
if (LUDecomposition(a,indx,3,&d))
|
|
{
|
|
VectorSet(forwardInv, 1.0, 0.0, 0.0);
|
|
BackSub(a,indx,forwardInv,3);
|
|
VectorSet(rightInv, 0.0, 1.0, 0.0);
|
|
BackSub(a,indx,rightInv,3);
|
|
VectorSet(upInv, 0.0, 0.0, 1.0);
|
|
BackSub(a,indx,upInv,3);
|
|
}
|
|
else
|
|
{
|
|
VectorClear(forwardInv);
|
|
VectorClear(rightInv);
|
|
VectorClear(upInv);
|
|
}
|
|
AngleVectors(pusher->s.angles, forward, right, up);
|
|
#else
|
|
VectorNegate(amove, org);
|
|
AngleVectors(org, forward, right, up);
|
|
#endif
|
|
|
|
// save the pusher's original position
|
|
pushed_p->ent = pusher;
|
|
|
|
VectorCopy(pusher->s.origin, pushed_p->origin);
|
|
VectorCopy(pusher->s.angles, pushed_p->angles);
|
|
|
|
if(pusher->client)
|
|
{
|
|
pushed_p->deltayaw = pusher->client->ps.pmove.delta_angles[YAW];
|
|
}
|
|
|
|
pushed_p++;
|
|
|
|
// move the pusher to it's final position
|
|
#if GIL_PUSH
|
|
VectorCopy(pusher->s.origin,OrgOrg);
|
|
#endif
|
|
Vec3AddAssign(move, pusher->s.origin);
|
|
Vec3AddAssign(amove, pusher->s.angles);
|
|
|
|
gi.linkentity(pusher);
|
|
|
|
// see if any solid entities are inside the final position
|
|
check = g_edicts+1;
|
|
|
|
for(e = 1; e < globals.num_edicts; ++e, ++check)
|
|
{
|
|
if(!check->inuse)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
switch(check->movetype)
|
|
{
|
|
case PHYSICSTYPE_PUSH:
|
|
case PHYSICSTYPE_STOP:
|
|
case PHYSICSTYPE_NONE:
|
|
case PHYSICSTYPE_NOCLIP:
|
|
continue;
|
|
}
|
|
|
|
// if the entity is standing on the pusher, it will definitely be moved
|
|
if(check->groundentity != pusher)
|
|
{
|
|
// see if the self needs to be tested
|
|
if(check->absmin[0] >= maxs[0]
|
|
|| check->absmin[1] >= maxs[1]
|
|
|| check->absmin[2] >= maxs[2]
|
|
|| check->absmax[0] <= mins[0]
|
|
|| check->absmax[1] <= mins[1]
|
|
|| check->absmax[2] <= mins[2])
|
|
continue;
|
|
|
|
// see if the self's bbox is inside the pusher's final position
|
|
if(!TestEntityPosition(check))
|
|
{
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if((pusher->movetype == PHYSICSTYPE_PUSH) || (check->groundentity == pusher))
|
|
{
|
|
// move this entity
|
|
pushed_p->ent = check;
|
|
|
|
VectorCopy(check->s.origin, pushed_p->origin);
|
|
VectorCopy(check->s.angles, pushed_p->angles);
|
|
VectorCopy(check->s.origin, holdOrg);
|
|
|
|
++pushed_p;
|
|
|
|
// try moving the contacted entity
|
|
|
|
if(check->client)
|
|
{ // FIXME: doesn't rotate monsters?
|
|
check->client->ps.pmove.delta_angles[YAW] += amove[YAW];
|
|
}
|
|
// figure movement due to the pusher's amove
|
|
#if GIL_PUSH
|
|
{
|
|
int test;
|
|
vec3_t testpnt;
|
|
for (test=0;test<4;test++)
|
|
{
|
|
VectorCopy(holdOrg,testpnt);
|
|
VectorCopy(holdOrg,check->s.origin);
|
|
testpnt[2]+=check->mins[2];
|
|
if (test&1)
|
|
testpnt[0]+=check->mins[0];
|
|
else
|
|
testpnt[0]+=check->maxs[0];
|
|
if (test&2)
|
|
testpnt[1]+=check->mins[1];
|
|
else
|
|
testpnt[1]+=check->maxs[1];
|
|
|
|
VectorSubtract(testpnt, OrgOrg, org);
|
|
Vec3AddAssign(move, check->s.origin);
|
|
org2[0] = DotProduct(org, forward);
|
|
org2[1] = -DotProduct(org, right);
|
|
org2[2] = DotProduct(org, up);
|
|
move2[0]=DotProduct(org2,forwardInv)-org[0];
|
|
move2[1]=DotProduct(org2,rightInv)-org[1];
|
|
move2[2]=DotProduct(org2,upInv)-org[2];
|
|
Vec3AddAssign(move2, check->s.origin);
|
|
|
|
// may have pushed them off an edge
|
|
if(check->groundentity != pusher)
|
|
{
|
|
check->groundentity = NULL;
|
|
}
|
|
|
|
block = TestEntityPosition(check);
|
|
if(!block)
|
|
break;
|
|
}
|
|
}
|
|
#else
|
|
Vec3AddAssign(move, check->s.origin);
|
|
VectorSubtract(check->s.origin, pusher->s.origin, org);
|
|
org2[0] = DotProduct(org, forward);
|
|
org2[1] = -DotProduct(org, right);
|
|
org2[2] = DotProduct(org, up);
|
|
VectorSubtract(org2, org, move2);
|
|
Vec3AddAssign(move2, check->s.origin);
|
|
|
|
// may have pushed them off an edge
|
|
if(check->groundentity != pusher)
|
|
{
|
|
check->groundentity = NULL;
|
|
}
|
|
|
|
block = TestEntityPosition(check);
|
|
#endif
|
|
|
|
if(!block)
|
|
{ // pushed ok
|
|
gi.linkentity (check);
|
|
|
|
if(check->client)
|
|
{
|
|
check->client->playerinfo.flags|=PLAYER_FLAG_USE_ENT_POS;
|
|
}
|
|
|
|
// impact?
|
|
continue;
|
|
}
|
|
// if it is ok to leave in the old position, do it
|
|
// this is only relevent for riding entities, not pushed
|
|
|
|
// No good, (A+B)-B!=A
|
|
//Vec3SubtractAssign(move, check->s.origin);
|
|
//Vec3SubtractAssign(move2, check->s.origin);
|
|
|
|
VectorCopy(holdOrg,check->s.origin);
|
|
block = TestEntityPosition(check);
|
|
|
|
if(!block)
|
|
{
|
|
--pushed_p;
|
|
continue;
|
|
}
|
|
}
|
|
|
|
// save off the obstacle so we can call the block function
|
|
obstacle = check;
|
|
|
|
// move back any entities we already moved
|
|
// go backwards, so if the same entity was pushed
|
|
// twice, it goes back to the original position
|
|
for(p = pushed_p - 1; p >= pushed; --p)
|
|
{
|
|
VectorCopy(p->origin, p->ent->s.origin);
|
|
VectorCopy(p->angles, p->ent->s.angles);
|
|
|
|
if(p->ent->client)
|
|
{
|
|
p->ent->client->ps.pmove.delta_angles[YAW] = p->deltayaw;
|
|
}
|
|
|
|
gi.linkentity(p->ent);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
//FIXME: is there a better way to handle this?
|
|
// see if anything we moved has touched a trigger
|
|
for(p = pushed_p - 1; p >= pushed; --p)
|
|
{
|
|
ActivateTriggers (p->ent);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
================
|
|
Physics_Push
|
|
|
|
Bmodel objects don't interact with each other, but
|
|
push all box objects
|
|
================
|
|
*/
|
|
void Physics_Push(edict_t *self)
|
|
{
|
|
vec3_t move, amove;
|
|
edict_t *part, *mv;
|
|
|
|
// Not a team captain, so movement will be handled elsewhere
|
|
if(self->flags & FL_TEAMSLAVE)
|
|
{
|
|
return;
|
|
}
|
|
|
|
// make sure all team slaves can move before commiting
|
|
// any moves or calling any think functions
|
|
// if the move is blocked, all moved objects will be backed out
|
|
pushed_p = pushed;
|
|
|
|
for(part = self; part; part = part->teamchain)
|
|
{
|
|
if(Vec3NotZero(part->velocity) || Vec3NotZero(part->avelocity))
|
|
{ // object is moving
|
|
VectorScale (part->velocity, FRAMETIME, move);
|
|
VectorScale (part->avelocity, FRAMETIME, amove);
|
|
|
|
if(!PushEntities(part, move, amove))
|
|
{
|
|
break; // move was blocked
|
|
}
|
|
}
|
|
}
|
|
|
|
if(pushed_p > &pushed[MAX_EDICTS])
|
|
{
|
|
gi.error (ERR_FATAL, "pushed_p > &pushed[MAX_EDICTS], memory corrupted");
|
|
}
|
|
|
|
if(part)
|
|
{
|
|
// the move failed, bump all nextthink times and back out moves
|
|
for(mv = self; mv; mv = mv->teamchain)
|
|
{
|
|
if(mv->nextthink > 0)
|
|
{
|
|
mv->nextthink += FRAMETIME;
|
|
}
|
|
}
|
|
|
|
// if the pusher has a "blocked" function, call it
|
|
// otherwise, just stay in place until the obstacle is gone
|
|
if(part->blocked)
|
|
{
|
|
part->blocked(part, obstacle);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// the move succeeded, so call all think functions
|
|
for(part = self; part; part = part->teamchain)
|
|
{
|
|
if(ThinkTime(part))
|
|
{
|
|
part->think(part);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static void Physics_ScriptAngular(edict_t *self)
|
|
{
|
|
vec3_t forward, dest, angle;
|
|
|
|
if (self->owner)
|
|
{
|
|
VectorAdd(self->owner->s.angles, self->moveinfo.start_angles, angle);
|
|
if (self->moveinfo.state & 1)
|
|
{
|
|
AngleVectors(angle, NULL, forward, NULL);
|
|
}
|
|
else
|
|
{
|
|
AngleVectors(angle, NULL, NULL, forward);
|
|
}
|
|
VectorInverse(forward);
|
|
VectorMA(self->moveinfo.start_origin, self->moveinfo.distance, forward, dest);
|
|
// Distance moved this frame
|
|
Vec3SubtractAssign(self->s.origin, dest);
|
|
// Hence velocity
|
|
VectorScale(dest, (1.0 / FRAMETIME), self->velocity);
|
|
|
|
if (self->moveinfo.state & 1)
|
|
{
|
|
VectorCopy(angle, self->s.angles);
|
|
}
|
|
}
|
|
Physics_Push(self);
|
|
}
|