From e1d991916cde5b2c01b0a729c0a76ffa5ba80267 Mon Sep 17 00:00:00 2001 From: Dominic Masters Date: Sun, 19 Mar 2023 10:17:33 -0700 Subject: [PATCH] About to implement physics --- src/dawn/physics/3d/AABB3D.cpp | 40 +++++++------- src/dawn/physics/3d/AABB3D.hpp | 5 ++ src/dawn/physics/3d/Ray3D.cpp | 1 - .../components/physics/3d/CMakeLists.txt | 1 + .../physics/3d/CharacterController3D.cpp | 35 +++++++++++++ .../physics/3d/CharacterController3D.hpp | 20 +++++++ .../scene/components/PlayerController.cpp | 52 +++++-------------- .../scene/components/PlayerController.hpp | 9 ++-- src/dawnrose/scenes/HelloWorldScene.hpp | 15 ++++-- 9 files changed, 109 insertions(+), 69 deletions(-) create mode 100644 src/dawn/scene/components/physics/3d/CharacterController3D.cpp create mode 100644 src/dawn/scene/components/physics/3d/CharacterController3D.hpp diff --git a/src/dawn/physics/3d/AABB3D.cpp b/src/dawn/physics/3d/AABB3D.cpp index 853e647a..b67a4d49 100644 --- a/src/dawn/physics/3d/AABB3D.cpp +++ b/src/dawn/physics/3d/AABB3D.cpp @@ -15,18 +15,18 @@ bool_t Dawn::aabb3dIntersect(struct AABB3D cube1, struct AABB3D cube2) { ); } -bool aabbSweep( - const glm::vec3& pos1, const glm::vec3& size1, const glm::vec3& vel1, - const glm::vec3& pos2, const glm::vec3& size2, const glm::vec3& vel2, - float& fraction +bool_t aabb3dSweep( + struct AABB3D cube1, glm::vec3 velocity1, + struct AABB3D cube2, glm::vec3 velocity2, + glm::vec3 *normal ) { glm::vec3 relVel = vel1 - vel2; glm::vec3 relPos = pos1 - pos2; // If the relative velocity is zero, the cubes are already intersecting - i (glm::length2(relVel) == 0.0f) { - fraction = 0.0f; - return true; + if(glm::length2(relVel) == 0.0f) { + fraction = 0.0f; + return true; } // Expand the size of the cubes by the magnitude of the relative velocity @@ -34,29 +34,29 @@ bool aabbSweep( glm::vec3 size2exp = size2 + glm::abs(relVel); // Compute the times at which the cubes first and last overlap on each axis - float tminx = (size1exp.x + size2exp.x - glm::abs(relPos.x)) / glm::abs(relVel.x); - float tmaxx = (size1.x + size2.x - glm::abs(relPos.x)) / glm::abs(relVel.x); - float tminy = (size1exp.y + size2exp.y - glm::abs(relPos.y)) / glm::abs(relVel.y); - float tmaxy = (size1.y + size2.y - glm::abs(relPos.y)) / glm::abs(relVel.y); - float tminz = (size1exp.z + size2exp.z - glm::abs(relPos.z)) / glm::abs(relVel.z); - float tmaxz = (size1.z + size2.z - glm::abs(relPos.z)) / glm::abs(relVel.z); + float_t tminx = (size1exp.x + size2exp.x - glm::abs(relPos.x)) / glm::abs(relVel.x); + float_t tmaxx = (size1.x + size2.x - glm::abs(relPos.x)) / glm::abs(relVel.x); + float_t tminy = (size1exp.y + size2exp.y - glm::abs(relPos.y)) / glm::abs(relVel.y); + float_t tmaxy = (size1.y + size2.y - glm::abs(relPos.y)) / glm::abs(relVel.y); + float_t tminz = (size1exp.z + size2exp.z - glm::abs(relPos.z)) / glm::abs(relVel.z); + float_t tmaxz = (size1.z + size2.z - glm::abs(relPos.z)) / glm::abs(relVel.z); // Find the earliest and latest times of overlap - float tmin = glm::max(glm::max(glm::min(tminx, tmaxx), glm::min(tminy, tmaxy)), glm::min(tminz, tmaxz)); - float tmax = glm::min(glm::min(glm::max(tminx, tmaxx), glm::max(tminy, tmaxy)), glm::max(tminz, tmaxz)); + float_t tmin = glm::max(glm::max(glm::min(tminx, tmaxx), glm::min(tminy, tmaxy)), glm::min(tminz, tmaxz)); + float_t tmax = glm::min(glm::min(glm::max(tminx, tmaxx), glm::max(tminy, tmaxy)), glm::max(tminz, tmaxz)); // If the earliest time of overlap is greater than the length of the timestep, the cubes // will not collide during the timestep if (tmin > 1.0f) { - fraction = 1.0f; - return false; + fraction = 1.0f; + return false; } // If the latest time of overlap is less than or equal to zero, the cubes are already // colliding and will collide throughout the timestep - if (tmax <= 0.0f) { - fraction = 0.0f; - return true; + if(tmax <= 0.0f) { + fraction = 0.0f; + return true; } // Compute the fraction of the timestep at which the collision occurs diff --git a/src/dawn/physics/3d/AABB3D.hpp b/src/dawn/physics/3d/AABB3D.hpp index b69b65e5..2e6fc780 100644 --- a/src/dawn/physics/3d/AABB3D.hpp +++ b/src/dawn/physics/3d/AABB3D.hpp @@ -22,4 +22,9 @@ namespace Dawn { bool_t aabb3dIntersect(struct AABB3D cube1, struct AABB3D cube2); + bool_t aabb3dSweep( + glm::vec3 pos1, glm::vec3 size1, glm::vec3 vel1, + glm::vec3 pos2, glm::vec3 size2, glm::vec3 vel2, + float_t& fraction + ); } \ No newline at end of file diff --git a/src/dawn/physics/3d/Ray3D.cpp b/src/dawn/physics/3d/Ray3D.cpp index 41b22af3..656fc195 100644 --- a/src/dawn/physics/3d/Ray3D.cpp +++ b/src/dawn/physics/3d/Ray3D.cpp @@ -29,7 +29,6 @@ bool_t Dawn::raytestSphere( *normal = glm::normalize(*hit - sphere.center); *distance = t0; - return true; } diff --git a/src/dawn/scene/components/physics/3d/CMakeLists.txt b/src/dawn/scene/components/physics/3d/CMakeLists.txt index 458e0e71..6a380f81 100644 --- a/src/dawn/scene/components/physics/3d/CMakeLists.txt +++ b/src/dawn/scene/components/physics/3d/CMakeLists.txt @@ -10,4 +10,5 @@ target_sources(${DAWN_TARGET_NAME} CubeCollider.cpp CapsuleCollider.cpp SphereCollider.cpp + CharacterController3D.cpp ) \ No newline at end of file diff --git a/src/dawn/scene/components/physics/3d/CharacterController3D.cpp b/src/dawn/scene/components/physics/3d/CharacterController3D.cpp new file mode 100644 index 00000000..ac43ef6b --- /dev/null +++ b/src/dawn/scene/components/physics/3d/CharacterController3D.cpp @@ -0,0 +1,35 @@ +// Copyright (c) 2023 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#include "CharacterController3D.hpp" + +using namespace Dawn; + +CharacterController3D::CharacterController3D(SceneItem *item) : + SceneItemComponent(item) +{ + +} + +void CharacterController3D::onStart() { + useEvent([&](float_t delta){ + // Friction + velocity -= glm::vec3(velocity.x, 0, velocity.z) * friction * delta; + + // Gravity + velocity += this->gravity * delta; + + auto myCollider = item->getComponent(); + auto colliders = getScene()->findComponents(); + auto itCollider = colliders.begin(); + while(itCollider != colliders.end()) { + (*itCollider)-> + ++itCollider; + } + + // Move / Update + transform->setLocalPosition(transform->getLocalPosition() + (velocity * delta)); + }, getScene()->eventSceneUpdate); +} \ No newline at end of file diff --git a/src/dawn/scene/components/physics/3d/CharacterController3D.hpp b/src/dawn/scene/components/physics/3d/CharacterController3D.hpp new file mode 100644 index 00000000..c8d8477d --- /dev/null +++ b/src/dawn/scene/components/physics/3d/CharacterController3D.hpp @@ -0,0 +1,20 @@ +// Copyright (c) 2023 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#pragma once +#include "Collider3D.hpp" + +namespace Dawn { + class CharacterController3D : public SceneItemComponent { + public: + glm::vec3 velocity = glm::vec3(0, 0, 0); + glm::vec3 gravity = glm::vec3(0, -1, 0); + float_t friction = 12.0f; + + CharacterController3D(SceneItem *item); + + void onStart() override; + }; +} \ No newline at end of file diff --git a/src/dawnrose/scene/components/PlayerController.cpp b/src/dawnrose/scene/components/PlayerController.cpp index 90adaa18..271699c5 100644 --- a/src/dawnrose/scene/components/PlayerController.cpp +++ b/src/dawnrose/scene/components/PlayerController.cpp @@ -14,13 +14,18 @@ PlayerController::PlayerController(SceneItem *item) : SceneItemComponent(item) { } -void PlayerController::onStart() { - useEvent([&](float_t delta){ - auto camera = getScene()->findComponent(); - assertNotNull(camera); +std::vector PlayerController::getDependencies() { + return { + (this->characterController = item->getComponent()) + }; +} - // Friction - velocity -= velocity * friction * delta; +void PlayerController::onStart() { + assertNotNull(this->characterController); + + useEvent([&](float_t delta){ + // auto camera = getScene()->findComponent(); + // assertNotNull(camera); // Movement auto inputMove = getGame()->inputManager.getAxis2D( @@ -31,40 +36,7 @@ void PlayerController::onStart() { if(inputMove.x != 0 || inputMove.y != 0) { float_t angle = atan2(inputMove.y, inputMove.x); glm::vec3 movement = glm::vec3(cos(angle), 0, sin(angle)); - velocity += movement * delta * moveSpeed; + characterController->velocity += movement * delta * moveSpeed; } - - // Move / Update - transform->setLocalPosition(transform->getLocalPosition() + (velocity * delta)); - - - // tEST - auto collider = item->getComponent(); - assertNotNull(collider); - Collider3DRayResult result; - - struct Ray3D ray; - ray.origin = glm::vec3(0, 0.0f, 5); - ray.direction = glm::vec3(0, 0.0f, -4); - - struct Color color = COLOR_MAGENTA; - - if(collider->raycast(&result, ray)) { - // std::cout << "Collide!" << std::endl; - color = COLOR_WHITE; - - getScene()->debugRay({ - .start = result.point, - .direction = result.normal, - .color = COLOR_GREEN - }); - } - - getScene()->debugRay({ - .start = ray.origin, - .direction = ray.direction, - .color = color - }); - }, getScene()->eventSceneUpdate); } \ No newline at end of file diff --git a/src/dawnrose/scene/components/PlayerController.hpp b/src/dawnrose/scene/components/PlayerController.hpp index faeb882c..74d1dc27 100644 --- a/src/dawnrose/scene/components/PlayerController.hpp +++ b/src/dawnrose/scene/components/PlayerController.hpp @@ -4,17 +4,20 @@ // https://opensource.org/licenses/MIT #pragma once -#include "scene/SceneItemComponent.hpp" +#include "scene/components/physics/3d/CharacterController3D.hpp" namespace Dawn { class PlayerController : public SceneItemComponent { + protected: + CharacterController3D *characterController; + public: - glm::vec3 velocity; float_t moveSpeed = 40.0f; - float_t friction = 12.0f; PlayerController(SceneItem *item); + std::vector getDependencies() override; + void onStart() override; }; } \ No newline at end of file diff --git a/src/dawnrose/scenes/HelloWorldScene.hpp b/src/dawnrose/scenes/HelloWorldScene.hpp index a0eb49c3..9c08335e 100644 --- a/src/dawnrose/scenes/HelloWorldScene.hpp +++ b/src/dawnrose/scenes/HelloWorldScene.hpp @@ -14,7 +14,9 @@ #include "scene/components/example/ExampleSpin.hpp" #include "scene/components/physics/3d/SphereCollider.hpp" #include "scene/components/physics/3d/CubeCollider.hpp" +#include "scene/components/physics/3d/CapsuleCollider.hpp" #include "display/mesh/SphereMesh.hpp" +#include "display/mesh/CapsuleMesh.hpp" namespace Dawn { class HelloWorldScene : public Scene { @@ -28,16 +30,19 @@ namespace Dawn { playerItem->addComponent(); auto meshHost = playerItem->addComponent(); playerItem->addComponent(); - auto hitbox = playerItem->addComponent(); - SphereMesh::createSphere(&meshHost->mesh, hitbox->radius, 32, 32); + auto hitbox = playerItem->addComponent(); + playerItem->addComponent(); + CapsuleMesh::create(&meshHost->mesh, hitbox->radius, hitbox->height); + + playerItem->transform.setLocalPosition(glm::vec3(0, 4, 0)); auto wall = this->createSceneItem(); auto wallHitbox = wall->addComponent(); - wall->transform.setLocalPosition(glm::vec3(-10, 0, 0)); - wallHitbox->min = -(wallHitbox->max = glm::vec3(1, 1, 5)); + // wall->transform.setLocalPosition(glm::vec3(-5, -1, -5)); + wallHitbox->min = -(wallHitbox->max = glm::vec3(5, 0.25f, 5)); camera = Camera::create(this); - camera->transform->lookAt(glm::vec3(4, 4, 4), glm::vec3(0, 0, 0)); + camera->transform->lookAt(glm::vec3(10, 10, 10), glm::vec3(0, 0, 0)); // auto gameCamera = camera->item->addComponent(); // gameCamera->player = player; }