Layers.h 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136
  1. // SPDX-FileCopyrightText: 2021 Jorrit Rouwe
  2. // SPDX-License-Identifier: MIT
  3. #pragma once
  4. #include <Jolt/Physics/Collision/ObjectLayer.h>
  5. #include <Jolt/Physics/Collision/BroadPhase/BroadPhaseLayer.h>
  6. /// Layer that objects can be in, determines which other objects it can collide with
  7. namespace Layers
  8. {
  9. static constexpr uint8 UNUSED1 = 0; // 5 unused values so that broadphase layers values don't match with object layer values (for testing purposes)
  10. static constexpr uint8 UNUSED2 = 1;
  11. static constexpr uint8 UNUSED3 = 2;
  12. static constexpr uint8 UNUSED4 = 3;
  13. static constexpr uint8 UNUSED5 = 4;
  14. static constexpr uint8 NON_MOVING = 5;
  15. static constexpr uint8 MOVING = 6;
  16. static constexpr uint8 HQ_DEBRIS = 7; // High quality debris collides with MOVING and NON_MOVING but not with any debris
  17. static constexpr uint8 LQ_DEBRIS = 8; // Low quality debris only collides with NON_MOVING
  18. static constexpr uint8 NUM_LAYERS = 9;
  19. };
  20. /// Class that determines if two object layers can collide
  21. class ObjectLayerPairFilterImpl : public ObjectLayerPairFilter
  22. {
  23. public:
  24. virtual bool ShouldCollide(ObjectLayer inObject1, ObjectLayer inObject2) const override
  25. {
  26. switch (inObject1)
  27. {
  28. case Layers::UNUSED1:
  29. case Layers::UNUSED2:
  30. case Layers::UNUSED3:
  31. case Layers::UNUSED4:
  32. case Layers::UNUSED5:
  33. return false;
  34. case Layers::NON_MOVING:
  35. return inObject2 == Layers::MOVING || inObject2 == Layers::HQ_DEBRIS || inObject2 == Layers::LQ_DEBRIS;
  36. case Layers::MOVING:
  37. return inObject2 == Layers::NON_MOVING || inObject2 == Layers::MOVING || inObject2 == Layers::HQ_DEBRIS;
  38. case Layers::HQ_DEBRIS:
  39. return inObject2 == Layers::NON_MOVING || inObject2 == Layers::MOVING;
  40. case Layers::LQ_DEBRIS:
  41. return inObject2 == Layers::NON_MOVING;
  42. default:
  43. JPH_ASSERT(false);
  44. return false;
  45. }
  46. }
  47. };
  48. /// Broadphase layers
  49. namespace BroadPhaseLayers
  50. {
  51. static constexpr BroadPhaseLayer NON_MOVING(0);
  52. static constexpr BroadPhaseLayer MOVING(1);
  53. static constexpr BroadPhaseLayer LQ_DEBRIS(2);
  54. static constexpr BroadPhaseLayer UNUSED(3);
  55. static constexpr uint NUM_LAYERS(4);
  56. };
  57. /// BroadPhaseLayerInterface implementation
  58. class BPLayerInterfaceImpl final : public BroadPhaseLayerInterface
  59. {
  60. public:
  61. BPLayerInterfaceImpl()
  62. {
  63. // Create a mapping table from object to broad phase layer
  64. mObjectToBroadPhase[Layers::UNUSED1] = BroadPhaseLayers::UNUSED;
  65. mObjectToBroadPhase[Layers::UNUSED2] = BroadPhaseLayers::UNUSED;
  66. mObjectToBroadPhase[Layers::UNUSED3] = BroadPhaseLayers::UNUSED;
  67. mObjectToBroadPhase[Layers::UNUSED4] = BroadPhaseLayers::UNUSED;
  68. mObjectToBroadPhase[Layers::UNUSED5] = BroadPhaseLayers::UNUSED;
  69. mObjectToBroadPhase[Layers::NON_MOVING] = BroadPhaseLayers::NON_MOVING;
  70. mObjectToBroadPhase[Layers::MOVING] = BroadPhaseLayers::MOVING;
  71. mObjectToBroadPhase[Layers::HQ_DEBRIS] = BroadPhaseLayers::MOVING; // HQ_DEBRIS is also in the MOVING layer as an example on how to map multiple layers onto the same broadphase layer
  72. mObjectToBroadPhase[Layers::LQ_DEBRIS] = BroadPhaseLayers::LQ_DEBRIS;
  73. }
  74. virtual uint GetNumBroadPhaseLayers() const override
  75. {
  76. return BroadPhaseLayers::NUM_LAYERS;
  77. }
  78. virtual BroadPhaseLayer GetBroadPhaseLayer(ObjectLayer inLayer) const override
  79. {
  80. JPH_ASSERT(inLayer < Layers::NUM_LAYERS);
  81. return mObjectToBroadPhase[inLayer];
  82. }
  83. #if defined(JPH_EXTERNAL_PROFILE) || defined(JPH_PROFILE_ENABLED)
  84. virtual const char * GetBroadPhaseLayerName(BroadPhaseLayer inLayer) const override
  85. {
  86. switch ((BroadPhaseLayer::Type)inLayer)
  87. {
  88. case (BroadPhaseLayer::Type)BroadPhaseLayers::NON_MOVING: return "NON_MOVING";
  89. case (BroadPhaseLayer::Type)BroadPhaseLayers::MOVING: return "MOVING";
  90. case (BroadPhaseLayer::Type)BroadPhaseLayers::LQ_DEBRIS: return "LQ_DEBRIS";
  91. case (BroadPhaseLayer::Type)BroadPhaseLayers::UNUSED: return "UNUSED";
  92. default: JPH_ASSERT(false); return "INVALID";
  93. }
  94. }
  95. #endif // JPH_EXTERNAL_PROFILE || JPH_PROFILE_ENABLED
  96. private:
  97. BroadPhaseLayer mObjectToBroadPhase[Layers::NUM_LAYERS];
  98. };
  99. /// Class that determines if an object layer can collide with a broadphase layer
  100. class ObjectVsBroadPhaseLayerFilterImpl : public ObjectVsBroadPhaseLayerFilter
  101. {
  102. public:
  103. virtual bool ShouldCollide(ObjectLayer inLayer1, BroadPhaseLayer inLayer2) const override
  104. {
  105. switch (inLayer1)
  106. {
  107. case Layers::NON_MOVING:
  108. return inLayer2 == BroadPhaseLayers::MOVING;
  109. case Layers::MOVING:
  110. case Layers::HQ_DEBRIS:
  111. return inLayer2 == BroadPhaseLayers::NON_MOVING || inLayer2 == BroadPhaseLayers::MOVING;
  112. case Layers::LQ_DEBRIS:
  113. return inLayer2 == BroadPhaseLayers::NON_MOVING;
  114. case Layers::UNUSED1:
  115. case Layers::UNUSED2:
  116. case Layers::UNUSED3:
  117. case Layers::UNUSED4:
  118. case Layers::UNUSED5:
  119. return false;
  120. default:
  121. JPH_ASSERT(false);
  122. return false;
  123. }
  124. }
  125. };