Working on different collider types.
This commit is contained in:
		| @@ -77,5 +77,23 @@ bool_t Dawn::boxCheckCollision( | |||||||
|   entryPoint = posA + velocity * entryTime; |   entryPoint = posA + velocity * entryTime; | ||||||
|   exitPoint = posA + velocity * exitTime; |   exitPoint = posA + velocity * exitTime; | ||||||
|  |  | ||||||
|  |   return true; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | bool_t Dawn::boxIsBoxColliding( | ||||||
|  |   glm::vec2 posA, glm::vec2 minA, glm::vec2 maxA, | ||||||
|  |   glm::vec2 posB, glm::vec2 minB, glm::vec2 maxB | ||||||
|  | ) { | ||||||
|  |   // Check for no overlap in X axis | ||||||
|  |   if (posA.x + maxA.x < posB.x + minB.x || posA.x + minA.x > posB.x + maxB.x) { | ||||||
|  |     return false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Check for no overlap in Y axis | ||||||
|  |   if (posA.y + maxA.y < posB.y + minB.y || posA.y + minA.y > posB.y + maxB.y) { | ||||||
|  |     return false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // There is overlap in both X and Y axis, so the boxes are colliding | ||||||
|   return true; |   return true; | ||||||
| } | } | ||||||
| @@ -20,6 +20,22 @@ namespace Dawn { | |||||||
|     glm::vec2 &exitPoint |     glm::vec2 &exitPoint | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Checks if two boxes are colliding. | ||||||
|  |    *  | ||||||
|  |    * @param posA Position of the first box. | ||||||
|  |    * @param minA Minimum point on the first box. | ||||||
|  |    * @param maxA Maximum point on the first box. | ||||||
|  |    * @param posB Position of the second box. | ||||||
|  |    * @param minB Minimum point on the second box. | ||||||
|  |    * @param maxB Maximum point on the second box. | ||||||
|  |    * @return True if the boxes are colliding with each other, false otherwise. | ||||||
|  |    */ | ||||||
|  |   bool_t boxIsBoxColliding( | ||||||
|  |     glm::vec2 posA, glm::vec2 minA, glm::vec2 maxA, | ||||||
|  |     glm::vec2 posB, glm::vec2 minB, glm::vec2 maxB | ||||||
|  |   ); | ||||||
|  |  | ||||||
|   /** |   /** | ||||||
|    * Checks if a given point is within the 2D Boundaries of an object. |    * Checks if a given point is within the 2D Boundaries of an object. | ||||||
|    *  |    *  | ||||||
|   | |||||||
| @@ -9,4 +9,6 @@ target_sources(${DAWN_TARGET_NAME} | |||||||
|     BoxCollider.cpp |     BoxCollider.cpp | ||||||
|     Collider2D.cpp |     Collider2D.cpp | ||||||
|     CharacterController2D.cpp |     CharacterController2D.cpp | ||||||
|  |     SolidController2D.cpp | ||||||
|  |     TriggerCollider2D.cpp | ||||||
| ) | ) | ||||||
| @@ -14,23 +14,23 @@ CharacterController2D::CharacterController2D(SceneItem *i) : | |||||||
|  |  | ||||||
| void CharacterController2D::onStart() { | void CharacterController2D::onStart() { | ||||||
|   useEvent([&](float_t delta){ |   useEvent([&](float_t delta){ | ||||||
|     // Friction |     // Common variables | ||||||
|     velocity -= velocity * friction * delta; |  | ||||||
|  |  | ||||||
|     auto myCollider = item->getComponent<Collider2D>(); |     auto myCollider = item->getComponent<Collider2D>(); | ||||||
|     glm::vec2 currentPosition( |     glm::vec2 currentPosition( | ||||||
|       this->transform->getLocalPosition().x, |       this->transform->getLocalPosition().x, | ||||||
|       this->transform->getLocalPosition().z |       this->transform->getLocalPosition().z | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|  |     // Friction | ||||||
|  |     velocity -= velocity * friction * delta; | ||||||
|  |  | ||||||
|     // Perform movement and check for collisions. |     // Solid Collision Check | ||||||
|     glm::vec2 moveAmount; |     glm::vec2 moveAmount; | ||||||
|     if(myCollider == nullptr) { |     if(myCollider == nullptr) { | ||||||
|       moveAmount = velocity; |       moveAmount = velocity; | ||||||
|     } else { |     } else { | ||||||
|       // Perform sweep |       // Perform sweep | ||||||
|       auto allColliders = getScene()->findComponents<Collider2D>(); |       auto allColliders = getScene()->findComponents<SolidController2D>(); | ||||||
|       auto itColliders = allColliders.begin(); |       auto itColliders = allColliders.begin(); | ||||||
|  |  | ||||||
|       struct CharacterController2DCollisionEventInfo info; |       struct CharacterController2DCollisionEventInfo info; | ||||||
| @@ -40,16 +40,17 @@ void CharacterController2D::onStart() { | |||||||
|       while(itColliders != allColliders.end()) { |       while(itColliders != allColliders.end()) { | ||||||
|         auto c = *itColliders; |         auto c = *itColliders; | ||||||
|         ++itColliders; |         ++itColliders; | ||||||
|         if(c == myCollider) continue; |         if(c->item == this->item) continue; | ||||||
|         result = myCollider->getCollidingResult( |         result = c->getCollidingResult( | ||||||
|           velocity, |           velocity, | ||||||
|           c, |           myCollider, | ||||||
|           info.normal, |           info.normal, | ||||||
|           info.entryTime, |           info.entryTime, | ||||||
|           info.exitTime, |           info.exitTime, | ||||||
|           info.entryPoint, |           info.entryPoint, | ||||||
|           info.exitPoint |           info.exitPoint | ||||||
|         ) && info.entryTime <= delta; |         ) && info.entryTime <= delta; | ||||||
|  |  | ||||||
|         if(result) { |         if(result) { | ||||||
|           info.collider = c; |           info.collider = c; | ||||||
|           break; |           break; | ||||||
| @@ -73,5 +74,18 @@ void CharacterController2D::onStart() { | |||||||
|     transform->setLocalPosition( |     transform->setLocalPosition( | ||||||
|       transform->getLocalPosition() + (glm::vec3(moveAmount.x, 0, moveAmount.y) * delta) |       transform->getLocalPosition() + (glm::vec3(moveAmount.x, 0, moveAmount.y) * delta) | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|  |     // Now perform trigger collision check | ||||||
|  |     auto allTriggers = getScene()->findComponents<TriggerController2D>(); | ||||||
|  |     auto itTriggers = allTriggers.begin(); | ||||||
|  |     while(itTriggers != allTriggers.end()) { | ||||||
|  |       auto c = *itTriggers; | ||||||
|  |       ++itTriggers; | ||||||
|  |       if(c->item == this->item) continue; | ||||||
|  |       if(c->getCollidingResult(myCollider)) { | ||||||
|  |         c->eventTriggerEnter.invoke(this); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|   }, getScene()->eventSceneUpdate); |   }, getScene()->eventSceneUpdate); | ||||||
| } | } | ||||||
| @@ -4,11 +4,12 @@ | |||||||
| // https://opensource.org/licenses/MIT | // https://opensource.org/licenses/MIT | ||||||
|  |  | ||||||
| #pragma once | #pragma once | ||||||
| #include "Collider2D.hpp" | #include "SolidController2D.hpp" | ||||||
|  | #include "TriggerController2D.hpp" | ||||||
|  |  | ||||||
| namespace Dawn { | namespace Dawn { | ||||||
|   struct CharacterController2DCollisionEventInfo { |   struct CharacterController2DCollisionEventInfo { | ||||||
|     Collider2D *collider; |     SolidController2D *collider; | ||||||
|     glm::vec2 normal; |     glm::vec2 normal; | ||||||
|     float_t entryTime; |     float_t entryTime; | ||||||
|     float_t exitTime; |     float_t exitTime; | ||||||
|   | |||||||
| @@ -10,58 +10,4 @@ using namespace Dawn; | |||||||
|  |  | ||||||
| Collider2D::Collider2D(SceneItem *i) : SceneItemComponent(i) { | Collider2D::Collider2D(SceneItem *i) : SceneItemComponent(i) { | ||||||
|    |    | ||||||
| } |  | ||||||
|  |  | ||||||
| bool_t Collider2D::getCollidingResult( |  | ||||||
|   glm::vec2 movement, |  | ||||||
|   Collider2D *other, |  | ||||||
|   glm::vec2 &normal, |  | ||||||
|   float_t &entryTime, |  | ||||||
|   float_t &exitTime, |  | ||||||
|   glm::vec2 &entryPoint, |  | ||||||
|   glm::vec2 &exitPoint |  | ||||||
| ) { |  | ||||||
|   assertNotNull(other); |  | ||||||
|   if(movement.x == 0 && movement.y == 0) return false; |  | ||||||
|  |  | ||||||
|   auto localPos = this->transform->getLocalPosition(); |  | ||||||
|   glm::vec2 myPos(localPos.x, localPos.z); |  | ||||||
|  |  | ||||||
|  |  | ||||||
|   // Check what THIS is |  | ||||||
|   switch(this->getColliderType()) { |  | ||||||
|     case COLLIDER2D_TYPE_BOX: { |  | ||||||
|       auto box1 = dynamic_cast<BoxCollider*>(this); |  | ||||||
|       assertNotNull(box1); |  | ||||||
|  |  | ||||||
|       // Box VS ? |  | ||||||
|       switch(other->getColliderType()) { |  | ||||||
|         case COLLIDER2D_TYPE_BOX: { |  | ||||||
|           auto box2 = dynamic_cast<BoxCollider*>(other); |  | ||||||
|           assertNotNull(box2); |  | ||||||
|           auto localPos2 = box2->transform->getLocalPosition(); |  | ||||||
|           glm::vec2 otherPos(localPos2.x, localPos2.z); |  | ||||||
|  |  | ||||||
|           return boxCheckCollision( |  | ||||||
|             myPos, box1->min, box1->max, |  | ||||||
|             otherPos, box2->min, box2->max, |  | ||||||
|             movement, |  | ||||||
|             normal, entryTime, exitTime, entryPoint, exitPoint |  | ||||||
|           ); |  | ||||||
|         } |  | ||||||
|          |  | ||||||
|         default: { |  | ||||||
|           assertUnreachable(); |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|       break; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     default: { |  | ||||||
|       assertUnreachable(); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|    |  | ||||||
|   assertUnreachable(); |  | ||||||
|   return false; |  | ||||||
| } | } | ||||||
| @@ -30,29 +30,5 @@ namespace Dawn { | |||||||
|        * @return The collider type that this is. |        * @return The collider type that this is. | ||||||
|        */ |        */ | ||||||
|       virtual enum Collider2DType getColliderType() = 0; |       virtual enum Collider2DType getColliderType() = 0; | ||||||
|  |  | ||||||
|       /** |  | ||||||
|        * Gets the result of checking if two collider bodies are intersecting. |  | ||||||
|        * This is performed WITH movement to return entry and exit times. |  | ||||||
|        *  |  | ||||||
|        * @param result Where to store the results of the calculation. |  | ||||||
|        * @param movement Movement operation that THIS object is performing. |  | ||||||
|        * @param other Other Collider that is being compared. |  | ||||||
|        * @param normal Output normal of the intersection. |  | ||||||
|        * @param entryTime Output entry time when the two objects will intersect. |  | ||||||
|        * @param exitTime Output exit time when the object will pass through. |  | ||||||
|        * @param entryPoint Output point in 2D space where object will enter. |  | ||||||
|        * @param exitPoint Output point where object will have passed through. |  | ||||||
|        * @return True if the two objects will intersect at move otherwise false. |  | ||||||
|        */ |  | ||||||
|       bool_t getCollidingResult( |  | ||||||
|         glm::vec2 movement, |  | ||||||
|         Collider2D *other, |  | ||||||
|         glm::vec2 &normal, |  | ||||||
|         float_t &entryTime, |  | ||||||
|         float_t &exitTime, |  | ||||||
|         glm::vec2 &entryPoint, |  | ||||||
|         glm::vec2 &exitPoint |  | ||||||
|       ); |  | ||||||
|   }; |   }; | ||||||
| } | } | ||||||
							
								
								
									
										72
									
								
								src/dawn/scene/components/physics/2d/SolidController2D.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										72
									
								
								src/dawn/scene/components/physics/2d/SolidController2D.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,72 @@ | |||||||
|  | // Copyright (c) 2023 Dominic Masters | ||||||
|  | //  | ||||||
|  | // This software is released under the MIT License. | ||||||
|  | // https://opensource.org/licenses/MIT | ||||||
|  |  | ||||||
|  | #include "SolidController2D.hpp" | ||||||
|  |  | ||||||
|  | using namespace Dawn; | ||||||
|  |  | ||||||
|  | SolidController2D::SolidController2D(SceneItem *item) : SceneItemComponent(item) { | ||||||
|  | } | ||||||
|  |  | ||||||
|  | std::vector<SceneItemComponent*> SolidController2D::getDependencies() { | ||||||
|  |   return { | ||||||
|  |     (this->collider = item->getComponent<Collider2D>()) | ||||||
|  |   }; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | bool_t SolidController2D::getCollidingResult( | ||||||
|  |   glm::vec2 movement, | ||||||
|  |   Collider2D *movingObject, | ||||||
|  |   glm::vec2 &normal, | ||||||
|  |   float_t &entryTime, | ||||||
|  |   float_t &exitTime, | ||||||
|  |   glm::vec2 &entryPoint, | ||||||
|  |   glm::vec2 &exitPoint | ||||||
|  | ) { | ||||||
|  |   assertNotNull(this->collider); | ||||||
|  |   assertNotNull(movingObject); | ||||||
|  |   if(movement.x == 0 && movement.y == 0) return false; | ||||||
|  |  | ||||||
|  |   auto localPos = this->transform->getLocalPosition(); | ||||||
|  |   glm::vec2 myPos(localPos.x, localPos.z); | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   // Check what THIS is | ||||||
|  |   switch(movingObject->getColliderType()) { | ||||||
|  |     case COLLIDER2D_TYPE_BOX: { | ||||||
|  |       auto box1 = dynamic_cast<BoxCollider*>(this); | ||||||
|  |       assertNotNull(box1); | ||||||
|  |  | ||||||
|  |       // Box VS ? | ||||||
|  |       switch(collider->getColliderType()) { | ||||||
|  |         case COLLIDER2D_TYPE_BOX: { | ||||||
|  |           auto box2 = dynamic_cast<BoxCollider*>(collider); | ||||||
|  |           assertNotNull(box2); | ||||||
|  |           auto localPos2 = box2->transform->getLocalPosition(); | ||||||
|  |           glm::vec2 otherPos(localPos2.x, localPos2.z); | ||||||
|  |  | ||||||
|  |           return boxCheckCollision( | ||||||
|  |             myPos, box1->min, box1->max, | ||||||
|  |             otherPos, box2->min, box2->max, | ||||||
|  |             movement, | ||||||
|  |             normal, entryTime, exitTime, entryPoint, exitPoint | ||||||
|  |           ); | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         default: { | ||||||
|  |           assertUnreachable(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |       break; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     default: { | ||||||
|  |       assertUnreachable(); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   assertUnreachable(); | ||||||
|  |   return false; | ||||||
|  | } | ||||||
							
								
								
									
										42
									
								
								src/dawn/scene/components/physics/2d/SolidController2D.hpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								src/dawn/scene/components/physics/2d/SolidController2D.hpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | |||||||
|  | // Copyright (c) 2023 Dominic Masters | ||||||
|  | //  | ||||||
|  | // This software is released under the MIT License. | ||||||
|  | // https://opensource.org/licenses/MIT | ||||||
|  |  | ||||||
|  | #pragma once | ||||||
|  | #include "Collider2D.hpp" | ||||||
|  | #include "BoxCollider.hpp" | ||||||
|  |  | ||||||
|  | namespace Dawn { | ||||||
|  |   class SolidController2D : public SceneItemComponent { | ||||||
|  |     public: | ||||||
|  |       Collider2D *collider = nullptr; | ||||||
|  |  | ||||||
|  |       SolidController2D(SceneItem *item); | ||||||
|  |       std::vector<SceneItemComponent*> getDependencies() override; | ||||||
|  |  | ||||||
|  |       /** | ||||||
|  |        * Gets the result of checking if two collider bodies are intersecting. | ||||||
|  |        * This is performed WITH movement to return entry and exit times. | ||||||
|  |        *  | ||||||
|  |        * @param result Where to store the results of the calculation. | ||||||
|  |        * @param movement Movement operation that THIS object is performing. | ||||||
|  |        * @param movingObject Moving Object Collider that is being compared. | ||||||
|  |        * @param normal Output normal of the intersection. | ||||||
|  |        * @param entryTime Output entry time when the two objects will intersect. | ||||||
|  |        * @param exitTime Output exit time when the object will pass through. | ||||||
|  |        * @param entryPoint Output point in 2D space where object will enter. | ||||||
|  |        * @param exitPoint Output point where object will have passed through. | ||||||
|  |        * @return True if the two objects will intersect at move otherwise false. | ||||||
|  |        */ | ||||||
|  |       bool_t getCollidingResult( | ||||||
|  |         glm::vec2 movement, | ||||||
|  |         Collider2D *movingObject, | ||||||
|  |         glm::vec2 &normal, | ||||||
|  |         float_t &entryTime, | ||||||
|  |         float_t &exitTime, | ||||||
|  |         glm::vec2 &entryPoint, | ||||||
|  |         glm::vec2 &exitPoint | ||||||
|  |       ); | ||||||
|  |   }; | ||||||
|  | } | ||||||
							
								
								
									
										50
									
								
								src/dawn/scene/components/physics/2d/TriggerController2D.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								src/dawn/scene/components/physics/2d/TriggerController2D.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,50 @@ | |||||||
|  | // Copyright (c) 2023 Dominic Masters | ||||||
|  | //  | ||||||
|  | // This software is released under the MIT License. | ||||||
|  | // https://opensource.org/licenses/MIT | ||||||
|  |  | ||||||
|  | #include "TriggerController2D.hpp" | ||||||
|  |  | ||||||
|  | using namespace Dawn; | ||||||
|  |  | ||||||
|  | TriggerController2D::TriggerController2D(SceneItem *i) : SceneItemComponent(i) { | ||||||
|  |    | ||||||
|  | } | ||||||
|  |  | ||||||
|  | std::vector<SceneItemComponent*> TriggerController2D::getDependencies() { | ||||||
|  |   return { | ||||||
|  |     (this->collider = this->item->getComponent<Collider2D>()) | ||||||
|  |   }; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | bool_t TriggerController2D::getCollidingResult(Collider2D* movingObject) { | ||||||
|  |   assertNotNull(this->collider); | ||||||
|  |   assertNotNull(movingObject); | ||||||
|  |  | ||||||
|  |   switch(movingObject->getColliderType()) { | ||||||
|  |     case COLLIDER2D_TYPE_BOX: { | ||||||
|  |       auto box1 = dynamic_cast<BoxCollider*>(movingObject); | ||||||
|  |       assertNotNull(box1); | ||||||
|  |  | ||||||
|  |       // Box VS ? | ||||||
|  |       switch(collider->getColliderType()) { | ||||||
|  |         case COLLIDER2D_TYPE_BOX: { | ||||||
|  |           auto box2 = dynamic_cast<BoxCollider*>(collider); | ||||||
|  |           assertNotNull(box2); | ||||||
|  |           return boxIsBoxColliding( | ||||||
|  |             box1->transform->getWorldPosition(), box1->min, box1->max, | ||||||
|  |             box2->transform->getWorldPosition(), box2->min, box2->max | ||||||
|  |           ); | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         default: { | ||||||
|  |           assertUnreachable(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     default: { | ||||||
|  |       assertUnreachable(); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										25
									
								
								src/dawn/scene/components/physics/2d/TriggerController2D.hpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								src/dawn/scene/components/physics/2d/TriggerController2D.hpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | |||||||
|  | // Copyright (c) 2023 Dominic Masters | ||||||
|  | //  | ||||||
|  | // This software is released under the MIT License. | ||||||
|  | // https://opensource.org/licenses/MIT | ||||||
|  |  | ||||||
|  | #pragma once | ||||||
|  | #include "Collider2D.hpp" | ||||||
|  |  | ||||||
|  | namespace Dawn { | ||||||
|  |   class TriggerController2D : public SceneItemComponent { | ||||||
|  |     public: | ||||||
|  |       Collider2D *collider = nullptr; | ||||||
|  |  | ||||||
|  |       StateEvent<CharacterController2D*> eventTriggerEnter; | ||||||
|  |  | ||||||
|  |       TriggerController2D(SceneItem *i); | ||||||
|  |       std::vector<SceneItemComponent*> getDependencies() override; | ||||||
|  |  | ||||||
|  |       /** | ||||||
|  |        * Returns whether or not the given moving object is colliding with this | ||||||
|  |        * trigger collider. | ||||||
|  |        */ | ||||||
|  |       bool_t getCollidingResult(Collider2D* movingObject) | ||||||
|  |   }; | ||||||
|  | } | ||||||
		Reference in New Issue
	
	Block a user