NS/dev/tankefugl/source/mod/CollisionChecker.cpp

765 lines
24 KiB
C++
Raw Normal View History

//new collision replacemnet
#include <vector>
#include <memory>
#include "common/mathlib.h"
#include "common/const.h"
#include "common/com_model.h"
#include "pm_shared/pm_defs.h"
#include "common/vector_util.h"
#include "mod/AvHHulls.h"
#include "types.h"
#include "util/MathUtil.h"
#include "CollisionChecker.h"
using std::auto_ptr;
//-----------------------------------------------------------------------------
// PlaneTest classes - used by CollisionChecker for testing against hulls
//-----------------------------------------------------------------------------
class CollisionTest
{
public:
virtual ~CollisionTest(void) {}
virtual int GetPlanarIntersectionType(const mplane_t* plane) const = 0;
virtual bool GetAABBIntersection(const nspoint_t& mins, const nspoint_t& maxs) const = 0;
virtual auto_ptr<CollisionTest> RotateAndShift(const nspoint_t& forward, const nspoint_t& up, const nspoint_t& right, const nspoint_t& origin) const = 0;
virtual auto_ptr<CollisionTest> Shift(const nspoint_t& origin) const = 0;
};
class PointCollisionTest : public CollisionTest
{
public:
PointCollisionTest(const nspoint_t& point)
{ SetPoint(point); }
void SetPoint(const nspoint_t& point)
{ VectorCopy(point,point_); }
int GetPlanarIntersectionType(const mplane_t* plane) const
{
float d;
if(plane->type < 3) // first three plane types are axial by index
{
d = point_[plane->type]*plane->normal[plane->type] - plane->dist;
}
else
{
d = DotProduct(plane->normal,point_) - plane->dist;
}
return ((d < 0) ? CollisionChecker::INTERSECTION_RELATION_BACK : CollisionChecker::INTERSECTION_RELATION_FRONT);
}
bool GetAABBIntersection(const nspoint_t& mins, const nspoint_t& maxs) const
{
return (
(point_[0] >= mins[0]) &&
(point_[0] <= maxs[0]) &&
(point_[1] >= mins[1]) &&
(point_[1] <= maxs[1]) &&
(point_[2] >= mins[2]) &&
(point_[2] <= maxs[1])
);
}
auto_ptr<CollisionTest> RotateAndShift(const nspoint_t& forward, const nspoint_t& right, const nspoint_t& up, const nspoint_t& origin) const
{
return Shift(origin);
}
auto_ptr<CollisionTest> Shift(const nspoint_t& origin) const
{
nspoint_t new_origin;
VectorAdd(origin,point_,new_origin);
return auto_ptr<CollisionTest>(new PointCollisionTest(new_origin));
}
private:
nspoint_t point_;
};
//-----------------------------------------------------------------------------
class CylinderCollisionTest : public CollisionTest
{
public:
CylinderCollisionTest(const nspoint_t& up, const nspoint_t& base, const float radius, const float height) : point_test_(base)
{
VectorCopy(up,up_);
VectorNormalize(up_);
VectorCopy(base,base_);
radius_ = radius;
height_ = height;
}
int GetPlanarIntersectionType(const mplane_t* plane) const
{
nspoint_t in_plane,out_of_plane,point_to_check;
CrossProduct(up_,plane->normal,in_plane);
CrossProduct(up_,in_plane,out_of_plane); //perp to up and pointing directly out of plane
VectorNormalize(out_of_plane);
VectorMA(base_,radius_,out_of_plane,point_to_check);
point_test_.SetPoint(point_to_check);
int intersect = point_test_.GetPlanarIntersectionType(plane);
VectorMA(point_to_check,height_,up_,point_to_check);
point_test_.SetPoint(point_to_check);
intersect |= point_test_.GetPlanarIntersectionType(plane);
if(intersect != CollisionChecker::INTERSECTION_RELATION_BOTH)
{
VectorMA(base_,-radius_,out_of_plane,point_to_check);
point_test_.SetPoint(point_to_check);
intersect |= point_test_.GetPlanarIntersectionType(plane);
}
if(intersect != CollisionChecker::INTERSECTION_RELATION_BOTH)
{
VectorMA(point_to_check,height_,up_,point_to_check);
point_test_.SetPoint(point_to_check);
intersect |= point_test_.GetPlanarIntersectionType(plane);
}
return intersect;
}
bool GetAABBIntersection(const nspoint_t& mins, const nspoint_t& maxs) const
{
//special case of unrotated cylinder
if(up_[2] == 1.0f)
{
if( maxs[2] < base_[2] ||
mins[2] > base_[2] + height_ ||
maxs[0] < base_[0] - radius_ ||
mins[0] > base_[0] + radius_ ||
maxs[1] < base_[1] - radius_ ||
maxs[1] > base_[1] + radius_
) { return false; }
//TODO: tighten definition to use overlap of circle instead of box with circle inscribed
return true;
}
//TODO: handle general case by turning bbox into hull and testing against that hull
ASSERT(0 && "Need to handle case of rotated cylinder & AABB collision");
return false;
}
auto_ptr<CollisionTest> RotateAndShift(const nspoint_t& forward, const nspoint_t& right, const nspoint_t& up, const nspoint_t& origin) const
{
nspoint_t new_up, new_base;
TransformVector(up_,forward,right,up,new_up);
VectorAdd(base_,origin,new_base);
return auto_ptr<CollisionTest>(new CylinderCollisionTest(new_up,new_base,radius_,height_));
}
auto_ptr<CollisionTest> Shift(const nspoint_t& origin) const
{
nspoint_t new_base;
VectorAdd(base_,origin,new_base);
return auto_ptr<CollisionTest>(new CylinderCollisionTest(up_,new_base,radius_,height_));
}
private:
mutable PointCollisionTest point_test_;
nspoint_t up_;
nspoint_t base_;
float radius_;
float height_;
};
//-----------------------------------------------------------------------------
//TODO: determine if there's a faster way than scanning all 8 corners
class OBBCollisionTest : public CollisionTest
{
public:
OBBCollisionTest(const OBBox& box) : point_test_(box.mOrigin)
{
box_ = box;
nspoint_t offset;
//calculate eight corners of the box given input angles:
for(int index = 0; index < 8; ++index)
{
offset[0] = box.mExtents[0] * box.mAxis[0][0];
offset[0] += box.mExtents[1] * box.mAxis[1][0];
offset[0] += box.mExtents[2] * box.mAxis[2][0];
if(index % 2) //every other item
{ offset[0] *= -1; }
offset[1] = box.mExtents[0] * box.mAxis[0][1];
offset[1] += box.mExtents[1] * box.mAxis[1][1];
offset[1] += box.mExtents[2] * box.mAxis[2][1];
if((index / 2) % 2) //every other pair of items
{ offset[1] *= -1; }
offset[2] = box.mExtents[0] * box.mAxis[0][2];
offset[2] += box.mExtents[1] * box.mAxis[1][2];
offset[2] += box.mExtents[2] * box.mAxis[2][2];
if(index < 4) //4 false, then 4 true
{ offset[2] *= -1; }
VectorAdd(box.mOrigin,offset,corners[index]);
}
}
int GetPlanarIntersectionType(const mplane_t* plane) const
{
int intersection_type = CollisionChecker::INTERSECTION_RELATION_NONE;
for(int index = 0; index < 8; ++index)
{
point_test_.SetPoint(corners[index]);
intersection_type |= point_test_.GetPlanarIntersectionType(plane);
if(intersection_type == CollisionChecker::INTERSECTION_RELATION_BOTH)
{ break; }
}
return intersection_type;
}
bool GetAABBIntersection(const nspoint_t& mins, const nspoint_t& maxs) const
{
bool intersection = false;
for(int index = 0; index < 8; ++index)
{
point_test_.SetPoint(corners[index]);
intersection = point_test_.GetAABBIntersection(mins,maxs);
if(intersection)
{ break; }
}
return intersection;
}
auto_ptr<CollisionTest> RotateAndShift(const nspoint_t& forward, const nspoint_t& right, const nspoint_t& up, const nspoint_t& origin) const
{
//make this an OBB by rotation around the ent angles - not a cheap operation.
OBBox box = box_;
VectorAdd(box_.mOrigin,origin,box.mOrigin);
TransformVector(box_.mAxis[0],forward,right,up,box.mAxis[0]);
TransformVector(box_.mAxis[1],forward,right,up,box.mAxis[1]);
TransformVector(box_.mAxis[2],forward,right,up,box.mAxis[2]);
//rotate axes!
return auto_ptr<CollisionTest>(new OBBCollisionTest(box));
}
auto_ptr<CollisionTest> Shift(const nspoint_t& origin) const
{
//make this an OBB by rotation around the ent angles - not a cheap operation.
OBBox box = box_;
VectorAdd(box_.mOrigin,origin,box.mOrigin);
return auto_ptr<CollisionTest>(new OBBCollisionTest(box));
}
private:
OBBox box_;
nspoint_t corners[8];
mutable PointCollisionTest point_test_;
};
//-----------------------------------------------------------------------------
class AABBCollisionTest : public CollisionTest
{
public:
AABBCollisionTest(const nspoint_t& mins, const nspoint_t& maxs) : point_test_(mins)
{
VectorCopy(maxs,maxs_);
VectorCopy(mins,mins_);
}
int GetPlanarIntersectionType(const mplane_t* plane) const
{
nspoint_t orientedMins = {
plane->normal[0] < 0 ? maxs_[0] : mins_[0],
plane->normal[1] < 0 ? maxs_[1] : mins_[1],
plane->normal[2] < 0 ? maxs_[2] : mins_[2]
};
point_test_.SetPoint(orientedMins);
int intersect_type = point_test_.GetPlanarIntersectionType(plane);
nspoint_t orientedMaxes = {
plane->normal[0] < 0 ? mins_[0] : maxs_[0],
plane->normal[1] < 0 ? mins_[1] : maxs_[1],
plane->normal[2] < 0 ? mins_[2] : maxs_[2]
};
point_test_.SetPoint(orientedMaxes);
intersect_type |= point_test_.GetPlanarIntersectionType(plane);
return intersect_type;
}
bool GetAABBIntersection(const nspoint_t& mins, const nspoint_t& maxs) const
{
return (
(maxs_[0] >= mins[0]) &&
(mins_[0] <= maxs[0]) &&
(maxs_[1] >= mins[1]) &&
(mins_[1] <= maxs[1]) &&
(maxs_[2] >= mins[2]) &&
(mins_[2] <= maxs[2])
);
}
auto_ptr<CollisionTest> RotateAndShift(const nspoint_t& forward, const nspoint_t& right, const nspoint_t& up, const nspoint_t& origin) const
{
//make this an OBB by rotation around the ent angles - not a cheap operation.
OBBox box;
//set up origin = ((mins + maxs) / 2) - ent origin
VectorCopy(mins_,box.mOrigin);
VectorAdd(maxs_,box.mOrigin,box.mOrigin);
VectorScale(box.mOrigin,0.5f,box.mOrigin);
VectorSubtract(box.mOrigin,origin,box.mOrigin);
//set up extents = ((maxs - mins) / 2)
VectorCopy(maxs_,box.mExtents);
VectorSubtract(box.mExtents,mins_,box.mExtents);
VectorScale(box.mExtents,0.5f,box.mExtents);
//set up axis = axis defined by ent angles
nspoint_t axis[3] = {{1,0,0},{0,1,0},{0,0,1}};
VectorCopy(forward,box.mAxis[0]);
VectorCopy(up,box.mAxis[1]);
VectorCopy(right,box.mAxis[2]);
return auto_ptr<CollisionTest>(new OBBCollisionTest(box));
}
auto_ptr<CollisionTest> Shift(const nspoint_t& origin) const
{
nspoint_t new_mins, new_maxs;
VectorAdd(origin,mins_,new_mins);
VectorAdd(origin,maxs_,new_maxs);
return auto_ptr<CollisionTest>(new AABBCollisionTest(new_mins,new_maxs));
}
private:
nspoint_t mins_, maxs_;
mutable PointCollisionTest point_test_;
};
//-----------------------------------------------------------------------------
// CollisionChecker class - making a fresh start on collision checking code
// that runs properly on both the server and client; requires that pmove is
// filled to function -- won't function fully for initial frames
//
// TODO: write quick functions for when pmove isn't available
// KGP - 4/25/04
//-----------------------------------------------------------------------------
const int CollisionChecker::NO_ENTITY = -1;
const int CollisionChecker::WORLD_ENTITY = 0;
const bool CollisionChecker::IGNORE_PLAYERS_FALSE = false;
const bool CollisionChecker::IGNORE_PLAYERS_TRUE = true;
const int CollisionChecker::HULL_TYPE_BBOX = 1;
const int CollisionChecker::HULL_TYPE_BSP = 2;
const int CollisionChecker::HULL_TYPE_ALL = 3;
const int CollisionChecker::INTERSECTION_RELATION_NONE = 0;
const int CollisionChecker::INTERSECTION_RELATION_FRONT = 1;
const int CollisionChecker::INTERSECTION_RELATION_BACK = 2;
const int CollisionChecker::INTERSECTION_RELATION_BOTH = 3;
CollisionChecker::CollisionChecker(const playermove_t* pmove) : pmove_(pmove)
{
SetHullNumber(0);
SetHullTypeMask(HULL_TYPE_ALL);
SetIgnorePlayers(false);
SetIgnoreEntityClass(IGNORE_NONE);
ClearEntityToIgnore();
}
CollisionChecker::CollisionChecker(const playermove_t* pmove, int ns_hull_number, int hull_type_mask, bool ignore_players, int ignore_entity_class, int entity_index_to_ignore) : pmove_(pmove)
{
SetHullNumber(ns_hull_number);
SetHullTypeMask(hull_type_mask);
SetIgnorePlayers(ignore_players);
SetIgnoreEntityClass(ignore_entity_class);
SetEntityToIgnore(entity_index_to_ignore);
}
//-------------------------------------------------------------------
// Configuration
//-------------------------------------------------------------------
void CollisionChecker::SetHullNumber(int ns_hull_number)
{
switch(ns_hull_number)
{
case kHLPointHullIndex:
valve_hull_number_ = 0;
break;
case kHLCrouchHullIndex:
valve_hull_number_ = 1;
break;
case kHLStandHullIndex:
valve_hull_number_ = 2;
break;
case kNSHugeHullIndex:
valve_hull_number_ = 3;
break;
default:
ASSERT(0 && "unknown hull index passed to CollisionChecker::SetHullNumber");
valve_hull_number_ = 0;
break;
}
}
//-------------------------------------------------------------------
void CollisionChecker::SetHullTypeMask(int hull_type_mask)
{
hull_type_mask_ = hull_type_mask;
}
//-------------------------------------------------------------------
void CollisionChecker::SetEntityToIgnore(int entity_index_to_ignore)
{
ASSERT(entity_index_to_ignore > 0 || entity_index_to_ignore == NO_ENTITY);
entity_index_to_ignore_ = entity_index_to_ignore;
}
//-------------------------------------------------------------------
void CollisionChecker::ClearEntityToIgnore(void)
{
entity_index_to_ignore_ = NO_ENTITY;
}
//-------------------------------------------------------------------
void CollisionChecker::SetIgnorePlayers(bool ignore_players)
{
ignore_players_ = ignore_players;
}
//-------------------------------------------------------------------
void CollisionChecker::SetIgnoreEntityClass(int ignore_entity_class)
{
ignore_entity_class_ = ignore_entity_class;
}
//-------------------------------------------------------------------
// Public point routines
//-------------------------------------------------------------------
int CollisionChecker::GetContentsAtPoint(const nspoint_t& point) const
{
return GetContents(&PointCollisionTest(point));
}
//-------------------------------------------------------------------
int CollisionChecker::GetWorldContentsAtPoint(const nspoint_t& point) const
{
return GetSingleEntityContentsAtPoint(point, WORLD_ENTITY);
}
//-------------------------------------------------------------------
int CollisionChecker::GetAllEntityContentsAtPoint(const nspoint_t& point) const
{
return GetAllEntityContents(&PointCollisionTest(point));
}
//-------------------------------------------------------------------
int CollisionChecker::GetSingleEntityContentsAtPoint(const nspoint_t& point, int entity_index) const
{
return GetSingleEntityContents(&PointCollisionTest(point),entity_index);
}
//-------------------------------------------------------------------
// Public Cylinder routines
//-------------------------------------------------------------------
const static nspoint_t CYLINDER_UP_DEFAULT = {0,0,1.0f};
int CollisionChecker::GetContentsInCylinder(const nspoint_t& base, float radius, float height) const
{
return GetContents(&CylinderCollisionTest(CYLINDER_UP_DEFAULT,base,radius,height));
}
//-------------------------------------------------------------------
int CollisionChecker::GetWorldContentsInCylinder(const nspoint_t& base, float radius, float height) const
{
return GetSingleEntityContents(&CylinderCollisionTest(CYLINDER_UP_DEFAULT,base,radius,height),WORLD_ENTITY);
}
//-------------------------------------------------------------------
int CollisionChecker::GetAllEntityContentsInCylinder(const nspoint_t& base, float radius, float height) const
{
return GetAllEntityContents(&CylinderCollisionTest(CYLINDER_UP_DEFAULT,base,radius,height));
}
//-------------------------------------------------------------------
int CollisionChecker::GetSingleEntityContentsInCylinder(const nspoint_t& base, float radius, float height, int entity_index) const
{
return GetSingleEntityContents(&CylinderCollisionTest(CYLINDER_UP_DEFAULT,base,radius,height),entity_index);
}
//-------------------------------------------------------------------
// Public AABB routines
//-------------------------------------------------------------------
int CollisionChecker::GetContentsInAABB(const nspoint_t& mins, const nspoint_t& maxs) const
{
return GetContents(&AABBCollisionTest(mins,maxs));
}
//-------------------------------------------------------------------
int CollisionChecker::GetWorldContentsInAABB(const nspoint_t& mins, const nspoint_t& maxs) const
{
return GetSingleEntityContents(&AABBCollisionTest(mins,maxs),WORLD_ENTITY);
}
//-------------------------------------------------------------------
int CollisionChecker::GetAllEntityContentsInAABB(const nspoint_t& mins, const nspoint_t& maxs) const
{
return GetAllEntityContents(&AABBCollisionTest(mins,maxs));
}
//-------------------------------------------------------------------
int CollisionChecker::GetSingleEntityContentsInAABB(const nspoint_t& mins, const nspoint_t& maxs, int entity_index) const
{
return GetSingleEntityContents(&AABBCollisionTest(mins,maxs),entity_index);
}
//-------------------------------------------------------------------
// Private utility routines
//-------------------------------------------------------------------
int CollisionChecker::CombineContents(const int old_contents, const int new_contents) const
{
if(new_contents == CONTENTS_EMPTY)
{ return old_contents; }
if(old_contents == CONTENT_EMPTY)
{ return new_contents; }
return max(old_contents,new_contents);
}
//-------------------------------------------------------------------
// Private content check routines
//-------------------------------------------------------------------
int CollisionChecker::GetContents(const CollisionTest* test) const
{
int total_contents = GetSingleEntityContents(test,WORLD_ENTITY);
if(total_contents != CONTENTS_SOLID)
{
total_contents = CombineContents(total_contents,GetAllEntityContents(test));
}
return total_contents;
}
//-------------------------------------------------------------------
int CollisionChecker::GetAllEntityContents(const CollisionTest* test) const
{
int total_contents = CONTENTS_EMPTY;
UpdateNumEntities();
for(int counter = WORLD_ENTITY+1; counter < entity_count_; ++counter)
{
total_contents = CombineContents(total_contents,GetSingleEntityContents(test,counter));
if(total_contents == CONTENTS_SOLID)
{ break; }
}
return total_contents;
}
//-------------------------------------------------------------------
int CollisionChecker::GetSingleEntityContents(const CollisionTest* test, int entity_index) const
{
int total_contents = CONTENTS_EMPTY;
const physent_t* ent = GetPhysentForIndex(entity_index);
if(ent && ent->info != entity_index_to_ignore_ && (!ignore_players_ || ent->info == WORLD_ENTITY || ent->info > 32))
{
if(ent->solid == SOLID_BBOX)
{
nspoint_t bbox_mins, bbox_maxs;
VectorSubtract(ent->mins,ent->origin,bbox_mins);
VectorSubtract(ent->maxs,ent->origin,bbox_maxs);
if(valve_hull_number_) //adjust for input hull
{
const hull_t* hull = &(&pmove_->physents[WORLD_ENTITY])->model->hulls[valve_hull_number_];
VectorAdd(bbox_mins,hull->clip_mins,bbox_mins);
VectorAdd(bbox_mins,hull->clip_maxs,bbox_maxs);
}
if(test->GetAABBIntersection(bbox_mins,bbox_maxs))
{
total_contents = CONTENTS_SOLID;
}
}
else if(ent->model && ent->model->type == mod_brush)
{
const hull_t* hull = &ent->model->hulls[valve_hull_number_];
if(ent->angles[0] || ent->angles[1] || ent->angles[2])
{
float axes[3][3];
AngleVectors(ent->angles,axes[0],axes[1],axes[2]);
auto_ptr<CollisionTest> rotated_test = test->RotateAndShift(axes[0],axes[1],axes[2],ent->origin);
total_contents = GetHullContent(hull,hull->firstclipnode,rotated_test.get());
}
else
{
auto_ptr<CollisionTest> shifted_test = test->Shift(ent->origin);
total_contents = GetHullContent(hull,hull->firstclipnode,shifted_test.get());
}
}
}
return total_contents;
}
//-------------------------------------------------------------------
//DFS search for solid hull leaf with passed in intersection test and short circuit on solid leaf found
int CollisionChecker::GetHullContent(const hull_t* hull, int current_node, const CollisionTest* test) const
{
ASSERT(current_node >= hull->firstclipnode);
ASSERT(current_node <= hull->lastclipnode);
int total_contents = CONTENTS_EMPTY;
if(current_node < 0)
{ total_contents = current_node; }
else
{
dclipnode_t *node = &hull->clipnodes[current_node];
mplane_t *plane;
int intersection;
while(true)
{
ASSERT(total_contents < 0); // 0 + is a node number and means we've had a bad assignment
//short circuit on solid found
if(total_contents == CONTENTS_SOLID)
{
this->intersected_nodes_.clear();
break;
}
plane = &hull->planes[node->planenum];
intersection = test->GetPlanarIntersectionType(&hull->planes[node->planenum]);
switch(intersection)
{
case INTERSECTION_RELATION_BOTH:
if(node->children[0] > CONTENTS_EMPTY) //non-leaf
{
if(node->children[1] > CONTENTS_EMPTY) //non-leaf
{
this->intersected_nodes_.push_back(node->children[0]); //store one path for later
node = &hull->clipnodes[node->children[1]]; //follow other path
continue;
}
total_contents = CombineContents(total_contents,node->children[1]);
node = &hull->clipnodes[node->children[0]];
continue;
}
else
{
total_contents = CombineContents(total_contents,node->children[0]);
if(node->children[1] > CONTENTS_EMPTY) //non-leaf
{
node = &hull->clipnodes[node->children[1]];
continue;
}
total_contents = CombineContents(total_contents,node->children[1]);
}
break;
case INTERSECTION_RELATION_FRONT:
if(node->children[0] > CONTENTS_EMPTY)
{
node = &hull->clipnodes[node->children[0]];
continue;
}
total_contents = CombineContents(total_contents,node->children[0]);
break;
case INTERSECTION_RELATION_BACK:
if(node->children[1] > CONTENTS_EMPTY)
{
node = &hull->clipnodes[node->children[1]];
continue;
}
total_contents = CombineContents(total_contents,node->children[1]);
break;
}
//check for other subtrees to process
if(this->intersected_nodes_.empty())
{ break; }
node = &hull->clipnodes[this->intersected_nodes_.back()];
this->intersected_nodes_.pop_back();
}
}
return total_contents;
}
//------------------------------------------------------------------------------
// Client/server specific code :( ... client version, server version in own file
//------------------------------------------------------------------------------
#ifdef AVH_CLIENT
const int CollisionChecker::IGNORE_NONE = 0;
const int CollisionChecker::IGNORE_INVISIBLE = 1;
const int CollisionChecker::IGNORE_INTANGIBLE = 2;
void CollisionChecker::UpdateNumEntities() const
{
switch(ignore_entity_class_)
{
case IGNORE_NONE:
case IGNORE_INVISIBLE:
entity_count_ = pmove_ ? pmove_->numvisent : 0;
break;
case IGNORE_INTANGIBLE:
entity_count_ = pmove_ ? pmove_->numphysent : 0;
break;
default:
ASSERT(0 && "Unknown ignore entity class");
};
}
const physent_t* CollisionChecker::GetPhysentForIndex(int entity_index) const
{
switch(ignore_entity_class_)
{
case IGNORE_NONE:
case IGNORE_INVISIBLE:
return pmove_ ? &pmove_->visents[entity_index] : (physent_t*)NULL;
case IGNORE_INTANGIBLE:
return pmove_ ? &pmove_->physents[entity_index] : (physent_t*)NULL;
}
ASSERT(0 && "Unknown ignore entity class");
return NULL;
}
#endif