Clip.h 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354
  1. /*
  2. ===========================================================================
  3. Doom 3 BFG Edition GPL Source Code
  4. Copyright (C) 1993-2012 id Software LLC, a ZeniMax Media company.
  5. This file is part of the Doom 3 BFG Edition GPL Source Code ("Doom 3 BFG Edition Source Code").
  6. Doom 3 BFG Edition Source Code is free software: you can redistribute it and/or modify
  7. it under the terms of the GNU General Public License as published by
  8. the Free Software Foundation, either version 3 of the License, or
  9. (at your option) any later version.
  10. Doom 3 BFG Edition Source Code is distributed in the hope that it will be useful,
  11. but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. GNU General Public License for more details.
  14. You should have received a copy of the GNU General Public License
  15. along with Doom 3 BFG Edition Source Code. If not, see <http://www.gnu.org/licenses/>.
  16. In addition, the Doom 3 BFG Edition Source Code is also subject to certain additional terms. You should have received a copy of these additional terms immediately following the terms and conditions of the GNU General Public License which accompanied the Doom 3 BFG Edition Source Code. If not, please request a copy in writing from id Software at the address below.
  17. If you have questions concerning this license or the applicable additional terms, you may contact in writing id Software LLC, c/o ZeniMax Media Inc., Suite 120, Rockville, Maryland 20850 USA.
  18. ===========================================================================
  19. */
  20. #ifndef __CLIP_H__
  21. #define __CLIP_H__
  22. /*
  23. ===============================================================================
  24. Handles collision detection with the world and between physics objects.
  25. ===============================================================================
  26. */
  27. #define CLIPMODEL_ID_TO_JOINT_HANDLE( id ) ( ( id ) >= 0 ? INVALID_JOINT : ((jointHandle_t) ( -1 - id )) )
  28. #define JOINT_HANDLE_TO_CLIPMODEL_ID( id ) ( -1 - id )
  29. class idClip;
  30. class idClipModel;
  31. class idEntity;
  32. //===============================================================
  33. //
  34. // idClipModel
  35. //
  36. //===============================================================
  37. class idClipModel {
  38. friend class idClip;
  39. public:
  40. idClipModel();
  41. explicit idClipModel( const char *name );
  42. explicit idClipModel( const idTraceModel &trm );
  43. explicit idClipModel( const idTraceModel &trm, bool persistantThroughSave );
  44. explicit idClipModel( const int renderModelHandle );
  45. explicit idClipModel( const idClipModel *model );
  46. ~idClipModel();
  47. bool LoadModel( const char *name );
  48. void LoadModel( const idTraceModel &trm, bool persistantThroughSave = true );
  49. void LoadModel( const int renderModelHandle );
  50. void Save( idSaveGame *savefile ) const;
  51. void Restore( idRestoreGame *savefile );
  52. void Link( idClip &clp ); // must have been linked with an entity and id before
  53. void Link( idClip &clp, idEntity *ent, int newId, const idVec3 &newOrigin, const idMat3 &newAxis, int renderModelHandle = -1 );
  54. void Unlink(); // unlink from sectors
  55. void SetPosition( const idVec3 &newOrigin, const idMat3 &newAxis ); // unlinks the clip model
  56. void Translate( const idVec3 &translation ); // unlinks the clip model
  57. void Rotate( const idRotation &rotation ); // unlinks the clip model
  58. void Enable(); // enable for clipping
  59. void Disable(); // keep linked but disable for clipping
  60. void SetMaterial( const idMaterial *m );
  61. const idMaterial * GetMaterial() const;
  62. void SetContents( int newContents ); // override contents
  63. int GetContents() const;
  64. void SetEntity( idEntity *newEntity );
  65. idEntity * GetEntity() const;
  66. void SetId( int newId );
  67. int GetId() const;
  68. void SetOwner( idEntity *newOwner );
  69. idEntity * GetOwner() const;
  70. const idBounds & GetBounds() const;
  71. const idBounds & GetAbsBounds() const;
  72. const idVec3 & GetOrigin() const;
  73. const idMat3 & GetAxis() const;
  74. bool IsTraceModel() const; // returns true if this is a trace model
  75. bool IsRenderModel() const; // returns true if this is a render model
  76. bool IsLinked() const; // returns true if the clip model is linked
  77. bool IsEnabled() const; // returns true if enabled for collision detection
  78. bool IsEqual( const idTraceModel &trm ) const;
  79. cmHandle_t Handle() const; // returns handle used to collide vs this model
  80. const idTraceModel * GetTraceModel() const;
  81. void GetMassProperties( const float density, float &mass, idVec3 &centerOfMass, idMat3 &inertiaTensor ) const;
  82. static cmHandle_t CheckModel( const char *name );
  83. static void ClearTraceModelCache();
  84. static int TraceModelCacheSize();
  85. static void SaveTraceModels( idSaveGame *savefile );
  86. static void RestoreTraceModels( idRestoreGame *savefile );
  87. private:
  88. bool enabled; // true if this clip model is used for clipping
  89. idEntity * entity; // entity using this clip model
  90. int id; // id for entities that use multiple clip models
  91. idEntity * owner; // owner of the entity that owns this clip model
  92. idVec3 origin; // origin of clip model
  93. idMat3 axis; // orientation of clip model
  94. idBounds bounds; // bounds
  95. idBounds absBounds; // absolute bounds
  96. const idMaterial * material; // material for trace models
  97. int contents; // all contents ored together
  98. cmHandle_t collisionModelHandle; // handle to collision model
  99. int traceModelIndex; // trace model used for collision detection
  100. int renderModelHandle; // render model def handle
  101. struct clipLink_s * clipLinks; // links into sectors
  102. int touchCount;
  103. void Init(); // initialize
  104. void Link_r( struct clipSector_s *node );
  105. static int AllocTraceModel( const idTraceModel &trm, bool persistantThroughSaves = true );
  106. static void FreeTraceModel( int traceModelIndex );
  107. static idTraceModel * GetCachedTraceModel( int traceModelIndex );
  108. static int GetTraceModelHashKey( const idTraceModel &trm );
  109. static struct trmCache_s * GetTraceModelEntry( int traceModelIndex );
  110. };
  111. ID_INLINE void idClipModel::Translate( const idVec3 &translation ) {
  112. Unlink();
  113. origin += translation;
  114. }
  115. ID_INLINE void idClipModel::Rotate( const idRotation &rotation ) {
  116. Unlink();
  117. origin *= rotation;
  118. axis *= rotation.ToMat3();
  119. }
  120. ID_INLINE void idClipModel::Enable() {
  121. enabled = true;
  122. }
  123. ID_INLINE void idClipModel::Disable() {
  124. enabled = false;
  125. }
  126. ID_INLINE void idClipModel::SetMaterial( const idMaterial *m ) {
  127. material = m;
  128. }
  129. ID_INLINE const idMaterial * idClipModel::GetMaterial() const {
  130. return material;
  131. }
  132. ID_INLINE void idClipModel::SetContents( int newContents ) {
  133. contents = newContents;
  134. }
  135. ID_INLINE int idClipModel::GetContents() const {
  136. return contents;
  137. }
  138. ID_INLINE void idClipModel::SetEntity( idEntity *newEntity ) {
  139. entity = newEntity;
  140. }
  141. ID_INLINE idEntity *idClipModel::GetEntity() const {
  142. return entity;
  143. }
  144. ID_INLINE void idClipModel::SetId( int newId ) {
  145. id = newId;
  146. }
  147. ID_INLINE int idClipModel::GetId() const {
  148. return id;
  149. }
  150. ID_INLINE void idClipModel::SetOwner( idEntity *newOwner ) {
  151. owner = newOwner;
  152. }
  153. ID_INLINE idEntity *idClipModel::GetOwner() const {
  154. return owner;
  155. }
  156. ID_INLINE const idBounds &idClipModel::GetBounds() const {
  157. return bounds;
  158. }
  159. ID_INLINE const idBounds &idClipModel::GetAbsBounds() const {
  160. return absBounds;
  161. }
  162. ID_INLINE const idVec3 &idClipModel::GetOrigin() const {
  163. return origin;
  164. }
  165. ID_INLINE const idMat3 &idClipModel::GetAxis() const {
  166. return axis;
  167. }
  168. ID_INLINE bool idClipModel::IsRenderModel() const {
  169. return ( renderModelHandle != -1 );
  170. }
  171. ID_INLINE bool idClipModel::IsTraceModel() const {
  172. return ( traceModelIndex != -1 );
  173. }
  174. ID_INLINE bool idClipModel::IsLinked() const {
  175. return ( clipLinks != NULL );
  176. }
  177. ID_INLINE bool idClipModel::IsEnabled() const {
  178. return enabled;
  179. }
  180. ID_INLINE bool idClipModel::IsEqual( const idTraceModel &trm ) const {
  181. return ( traceModelIndex != -1 && *GetCachedTraceModel( traceModelIndex ) == trm );
  182. }
  183. ID_INLINE const idTraceModel *idClipModel::GetTraceModel() const {
  184. if ( !IsTraceModel() ) {
  185. return NULL;
  186. }
  187. return idClipModel::GetCachedTraceModel( traceModelIndex );
  188. }
  189. //===============================================================
  190. //
  191. // idClip
  192. //
  193. //===============================================================
  194. class idClip {
  195. friend class idClipModel;
  196. public:
  197. idClip();
  198. void Init();
  199. void Shutdown();
  200. // clip versus the rest of the world
  201. bool Translation( trace_t &results, const idVec3 &start, const idVec3 &end,
  202. const idClipModel *mdl, const idMat3 &trmAxis, int contentMask, const idEntity *passEntity );
  203. bool Rotation( trace_t &results, const idVec3 &start, const idRotation &rotation,
  204. const idClipModel *mdl, const idMat3 &trmAxis, int contentMask, const idEntity *passEntity );
  205. bool Motion( trace_t &results, const idVec3 &start, const idVec3 &end, const idRotation &rotation,
  206. const idClipModel *mdl, const idMat3 &trmAxis, int contentMask, const idEntity *passEntity );
  207. int Contacts( contactInfo_t *contacts, const int maxContacts, const idVec3 &start, const idVec6 &dir, const float depth,
  208. const idClipModel *mdl, const idMat3 &trmAxis, int contentMask, const idEntity *passEntity );
  209. int Contents( const idVec3 &start,
  210. const idClipModel *mdl, const idMat3 &trmAxis, int contentMask, const idEntity *passEntity );
  211. // special case translations versus the rest of the world
  212. bool TracePoint( trace_t &results, const idVec3 &start, const idVec3 &end,
  213. int contentMask, const idEntity *passEntity );
  214. bool TraceBounds( trace_t &results, const idVec3 &start, const idVec3 &end, const idBounds &bounds,
  215. int contentMask, const idEntity *passEntity );
  216. // clip versus a specific model
  217. void TranslationModel( trace_t &results, const idVec3 &start, const idVec3 &end,
  218. const idClipModel *mdl, const idMat3 &trmAxis, int contentMask,
  219. cmHandle_t model, const idVec3 &modelOrigin, const idMat3 &modelAxis );
  220. void RotationModel( trace_t &results, const idVec3 &start, const idRotation &rotation,
  221. const idClipModel *mdl, const idMat3 &trmAxis, int contentMask,
  222. cmHandle_t model, const idVec3 &modelOrigin, const idMat3 &modelAxis );
  223. int ContactsModel( contactInfo_t *contacts, const int maxContacts, const idVec3 &start, const idVec6 &dir, const float depth,
  224. const idClipModel *mdl, const idMat3 &trmAxis, int contentMask,
  225. cmHandle_t model, const idVec3 &modelOrigin, const idMat3 &modelAxis );
  226. int ContentsModel( const idVec3 &start,
  227. const idClipModel *mdl, const idMat3 &trmAxis, int contentMask,
  228. cmHandle_t model, const idVec3 &modelOrigin, const idMat3 &modelAxis );
  229. // clip versus all entities but not the world
  230. void TranslationEntities( trace_t &results, const idVec3 &start, const idVec3 &end,
  231. const idClipModel *mdl, const idMat3 &trmAxis, int contentMask, const idEntity *passEntity );
  232. // get a contact feature
  233. bool GetModelContactFeature( const contactInfo_t &contact, const idClipModel *clipModel, idFixedWinding &winding ) const;
  234. // get entities/clip models within or touching the given bounds
  235. int EntitiesTouchingBounds( const idBounds &bounds, int contentMask, idEntity **entityList, int maxCount ) const;
  236. int ClipModelsTouchingBounds( const idBounds &bounds, int contentMask, idClipModel **clipModelList, int maxCount ) const;
  237. const idBounds & GetWorldBounds() const;
  238. idClipModel * DefaultClipModel();
  239. // stats and debug drawing
  240. void PrintStatistics();
  241. void DrawClipModels( const idVec3 &eye, const float radius, const idEntity *passEntity );
  242. bool DrawModelContactFeature( const contactInfo_t &contact, const idClipModel *clipModel, int lifetime ) const;
  243. private:
  244. int numClipSectors;
  245. struct clipSector_s * clipSectors;
  246. idBounds worldBounds;
  247. idClipModel temporaryClipModel;
  248. idClipModel defaultClipModel;
  249. mutable int touchCount;
  250. // statistics
  251. int numTranslations;
  252. int numRotations;
  253. int numMotions;
  254. int numRenderModelTraces;
  255. int numContents;
  256. int numContacts;
  257. private:
  258. struct clipSector_s * CreateClipSectors_r( const int depth, const idBounds &bounds, idVec3 &maxSector );
  259. void ClipModelsTouchingBounds_r( const struct clipSector_s *node, struct listParms_s &parms ) const;
  260. const idTraceModel * TraceModelForClipModel( const idClipModel *mdl ) const;
  261. int GetTraceClipModels( const idBounds &bounds, int contentMask, const idEntity *passEntity, idClipModel **clipModelList ) const;
  262. void TraceRenderModel( trace_t &trace, const idVec3 &start, const idVec3 &end, const float radius, const idMat3 &axis, idClipModel *touch ) const;
  263. };
  264. ID_INLINE bool idClip::TracePoint( trace_t &results, const idVec3 &start, const idVec3 &end, int contentMask, const idEntity *passEntity ) {
  265. Translation( results, start, end, NULL, mat3_identity, contentMask, passEntity );
  266. return ( results.fraction < 1.0f );
  267. }
  268. ID_INLINE bool idClip::TraceBounds( trace_t &results, const idVec3 &start, const idVec3 &end, const idBounds &bounds, int contentMask, const idEntity *passEntity ) {
  269. temporaryClipModel.LoadModel( idTraceModel( bounds ) );
  270. Translation( results, start, end, &temporaryClipModel, mat3_identity, contentMask, passEntity );
  271. return ( results.fraction < 1.0f );
  272. }
  273. ID_INLINE const idBounds & idClip::GetWorldBounds() const {
  274. return worldBounds;
  275. }
  276. ID_INLINE idClipModel *idClip::DefaultClipModel() {
  277. return &defaultClipModel;
  278. }
  279. #endif /* !__CLIP_H__ */