From 9100ab4d7eba374c7b2ea1c6ee4154be1181cebd Mon Sep 17 00:00:00 2001 From: Joseph Walton-Rivers <joseph@walton-rivers.uk> Date: Wed, 5 Jul 2023 16:49:51 +0100 Subject: [PATCH] basic native physics support --- CMakeLists.txt | 2 + components/CMakeLists.txt | 1 + components/physics/CMakeLists.txt | 11 ++ components/physics/docs/NOTICE | 31 +++++ components/physics/include/phys/module.hpp | 68 +++++++++++ components/physics/include/phys/types.hpp | 38 +++++++ components/physics/src/collision.cpp | 21 ++++ components/physics/src/integration.cpp | 41 +++++++ components/physics/src/service.cpp | 125 +++++++++++++++++++++ demo/CMakeLists.txt | 2 +- demo/demo/grid.cpp | 2 +- demo/demo/main.cpp | 4 +- include/fggl/entity/gridworld/zone.hpp | 4 +- include/fggl/grid/hexagon.hpp | 1 + include/fggl/math/easing.hpp | 8 +- include/fggl/math/fmath.hpp | 14 +-- 16 files changed, 356 insertions(+), 17 deletions(-) create mode 100644 components/CMakeLists.txt create mode 100644 components/physics/CMakeLists.txt create mode 100644 components/physics/docs/NOTICE create mode 100644 components/physics/include/phys/module.hpp create mode 100644 components/physics/include/phys/types.hpp create mode 100644 components/physics/src/collision.cpp create mode 100644 components/physics/src/integration.cpp create mode 100644 components/physics/src/service.cpp diff --git a/CMakeLists.txt b/CMakeLists.txt index 1c7eb76..7368b2f 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -48,6 +48,8 @@ add_subdirectory( fggl ) target_compile_options( fggl PRIVATE -Wall -Wpedantic -Wextra -Wodr -fno-strict-aliasing -fno-strict-overflow ) set_property(TARGET fggl PROPERTY INTERPROCEDURAL_OPTIMIZATION True) +add_subdirectory( components ) + # Unit Tests if (FGGL_TESTS) include(CTest) diff --git a/components/CMakeLists.txt b/components/CMakeLists.txt new file mode 100644 index 0000000..e220ee7 --- /dev/null +++ b/components/CMakeLists.txt @@ -0,0 +1 @@ +add_subdirectory(physics) \ No newline at end of file diff --git a/components/physics/CMakeLists.txt b/components/physics/CMakeLists.txt new file mode 100644 index 0000000..c9241e8 --- /dev/null +++ b/components/physics/CMakeLists.txt @@ -0,0 +1,11 @@ +add_library(fggl-phys) + +target_link_libraries(fggl-phys fggl) + +target_include_directories(fggl-phys PUBLIC include) +target_sources(fggl-phys + PRIVATE + src/collision.cpp + src/integration.cpp + src/service.cpp + ) \ No newline at end of file diff --git a/components/physics/docs/NOTICE b/components/physics/docs/NOTICE new file mode 100644 index 0000000..703fdb3 --- /dev/null +++ b/components/physics/docs/NOTICE @@ -0,0 +1,31 @@ +fggl-physics includes code derived from: + +Game Physics Engine Development, by Ian Millington +CRC Press, ISBN 978-0-12-381976-5 + +This book includes code from Cyclone Physics, available under the MIT licence: + +The MIT License + +Copyright (c) 2003-2009 Ian Millington + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE. + + + diff --git a/components/physics/include/phys/module.hpp b/components/physics/include/phys/module.hpp new file mode 100644 index 0000000..f770494 --- /dev/null +++ b/components/physics/include/phys/module.hpp @@ -0,0 +1,68 @@ +/* + * This file is part of FGGL. + * + * FGGL is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any + * later version. + * + * FGGL is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty + * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along with FGGL. + * If not, see <https://www.gnu.org/licenses/>. + */ + +// +// Created by webpigeon on 24/06/23. +// + +#ifndef FGGL_COMPONENTS_PHYSICS_INCLUDE_PHYS_MODULE_HPP +#define FGGL_COMPONENTS_PHYSICS_INCLUDE_PHYS_MODULE_HPP + +#include <fggl/phys/types.hpp> +#include <fggl/phys/service.hpp> + +namespace fggl::phys { + + class InternalPhysicsEngine : public PhysicsEngine { + public: + explicit InternalPhysicsEngine( entity::EntityManager* entities ); + void step() override; + + std::vector<ContactPoint> scanCollisions(entity::EntityID) override; + + entity::EntityID raycast(math::vec3, math::vec3) override; + std::vector<entity::EntityID> raycastAll(math::vec3, math::vec3) override; + std::vector<entity::EntityID> sweep(PhyShape &shape, + math::Transform &from, + math::Transform &to) override; + + inline void setDebugDraw(bool enable) override { + m_debugDraw = enable; + } + + private: + entity::EntityManager* m_entities; + bool m_debugDraw = false; + }; + + class InternalPhysicsProvider : public PhysicsProvider { + public: + virtual ~InternalPhysicsProvider() = default; + InternalPhysicsEngine* create(entity::EntityManager*, entity::EntityFactory*) override; + }; + + struct InternalPhysics { + constexpr static const char *name = "fggl::phys::internal"; + constexpr static const std::array<modules::ServiceName, 1> provides { + phys::PhysicsProvider::service + }; + constexpr static const std::array<modules::ServiceName, 1> depends { + entity::EntityFactory::service + }; + static bool factory(modules::ServiceName name, modules::Services& services); + }; + +} // namespace fggl::phys + +#endif //FGGL_COMPONENTS_PHYSICS_INCLUDE_PHYS_MODULE_HPP diff --git a/components/physics/include/phys/types.hpp b/components/physics/include/phys/types.hpp new file mode 100644 index 0000000..70a4853 --- /dev/null +++ b/components/physics/include/phys/types.hpp @@ -0,0 +1,38 @@ +/* + * This file is part of FGGL. + * + * FGGL is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any + * later version. + * + * FGGL is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty + * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along with FGGL. + * If not, see <https://www.gnu.org/licenses/>. + */ + +// +// Created by webpigeon on 24/06/23. +// + +#ifndef FGGL_COMPONENTS_PHYSICS_INCLUDE_PHYS_TYPES_HPP +#define FGGL_COMPONENTS_PHYSICS_INCLUDE_PHYS_TYPES_HPP + +#include <fggl/math/fmath.hpp> +#include <fggl/entity/module.hpp> + +namespace fggl::phys { + constexpr float FLOAT_HALF = 0.5F; + + struct Motion { + math::vec3f velocity; + math::vec3f acceleration; + float damping = 0.9F; + }; + + void euler(entity::EntityManager& entities, float duration); + +} // namespace fggl::phys + +#endif //FGGL_COMPONENTS_PHYSICS_INCLUDE_PHYS_TYPES_HPP diff --git a/components/physics/src/collision.cpp b/components/physics/src/collision.cpp new file mode 100644 index 0000000..0fae02d --- /dev/null +++ b/components/physics/src/collision.cpp @@ -0,0 +1,21 @@ +/* + * This file is part of FGGL. + * + * FGGL is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any + * later version. + * + * FGGL is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty + * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along with FGGL. + * If not, see <https://www.gnu.org/licenses/>. + */ + +namespace fggl::phys { + + void detect_collisions() { + + } + +} // namespace fggl::phys \ No newline at end of file diff --git a/components/physics/src/integration.cpp b/components/physics/src/integration.cpp new file mode 100644 index 0000000..a94c98a --- /dev/null +++ b/components/physics/src/integration.cpp @@ -0,0 +1,41 @@ +/* + * This file is part of FGGL. + * + * FGGL is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any + * later version. + * + * FGGL is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty + * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along with FGGL. + * If not, see <https://www.gnu.org/licenses/>. + */ + +#include <phys/types.hpp> +#include <fggl/entity/module.hpp> + +namespace fggl::phys { + + void euler(entity::EntityManager& entities, float duration) { + + auto physEntities = entities.find<math::Transform, Motion, Dynamics>(); + + for ( const auto& entity : physEntities ) { + auto& transform = entities.get<math::Transform>(entity); + auto& motion = entities.get<Motion>(entity); + auto& dynamics = entities.get<Dynamics>(entity); + + transform.origin( math::add_scaled(transform.origin(), motion.velocity, duration) ); + //math::add_scaled(entity.position, entity.acceleration, duration * duration * FLOAT_HALF); + + math::vec3f const resultingAcc = dynamics.force; + motion.velocity = math::add_scaled( motion.velocity, resultingAcc, duration ); + motion.velocity *= powf( motion.damping, duration); + + dynamics.force = math::VEC3_ZERO; + } + + } + +} // namespace fggl::phys \ No newline at end of file diff --git a/components/physics/src/service.cpp b/components/physics/src/service.cpp new file mode 100644 index 0000000..4cc4813 --- /dev/null +++ b/components/physics/src/service.cpp @@ -0,0 +1,125 @@ +/* + * This file is part of FGGL. + * + * FGGL is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any + * later version. + * + * FGGL is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty + * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along with FGGL. + * If not, see <https://www.gnu.org/licenses/>. + */ + +// +// Created by webpigeon on 24/06/23. +// + +#include <phys/module.hpp> +#include <phys/types.hpp> + +namespace fggl::phys { + + namespace { + void build_internal(const entity::ComponentSpec & /*config*/, + entity::EntityManager &manager, + const entity::EntityID &entity, + modules::Services &/*services*/) { + manager.add<fggl::phys::RigidBody>(entity); + manager.add<fggl::phys::Dynamics>(entity); + manager.add<fggl::phys::Motion>(entity); + } + } + + constexpr float DELTA_60 = 1000.0F / 60.0F; + constexpr float DELTA_TO_SECONDS = 1.F / 1000.F; + + InternalPhysicsEngine *InternalPhysicsProvider::create(entity::EntityManager *manager, entity::EntityFactory *factory) { + factory->bind(util::make_guid("phys::Body"), build_internal); + return new InternalPhysicsEngine(manager); + } + + bool InternalPhysics::factory(modules::ServiceName serviceName, modules::Services &services) { + if ( serviceName == phys::PhysicsProvider::service ) { + services.bind<phys::PhysicsProvider, InternalPhysicsProvider>(); + return true; + } + return false; + } + + void checkCollisions(entity::EntityManager& manager) { + auto colliders = manager.find<math::Transform, RigidBody>(); + for ( auto entity : colliders ) { + + } + } + + void InternalPhysicsEngine::step() { + euler( *m_entities, DELTA_60 * DELTA_TO_SECONDS ); + } + + + bool overlaps(const entity::EntityID& e1, math::Transform& t1, RigidBody& b1, + const entity::EntityID& e2, math::Transform& t2, RigidBody& b2, ContactPoint& cp) { + + auto distance = glm::distance(t1.origin(), t2.origin()); + if ( distance <= 2.0F ) { + cp.entityA = e1; + cp.entityB = e2; + + return true; + } + + return false; + } + + std::vector<ContactPoint> InternalPhysicsEngine::scanCollisions(entity::EntityID target) { + // really dumb N^2 version, need to use acceleration structure + + auto colliders = m_entities->find<math::Transform, RigidBody>(); + std::vector<ContactPoint> contacts; + + auto targetTransform = m_entities->get<math::Transform>(target); + auto targetBody = m_entities->get<RigidBody>(target); + + for ( const auto& entity : colliders ) { + // cannot collide with itself + if ( entity == target) { + continue; + } + + auto transform = m_entities->get<math::Transform>(entity); + auto body = m_entities->get<RigidBody>(entity); + + // check there is an overlap + ContactPoint contact{}; + if ( overlaps(target, targetTransform, targetBody, entity, transform, body, contact) ) { + contacts.push_back(contact); + } + + } + + return contacts; + } + + entity::EntityID InternalPhysicsEngine::raycast(math::vec3, math::vec3) { + entity::EntityID result; + return result; + } + + std::vector<entity::EntityID> InternalPhysicsEngine::raycastAll(math::vec3, math::vec3) { + return std::vector<entity::EntityID>(); + } + + std::vector<entity::EntityID> InternalPhysicsEngine::sweep(PhyShape &shape, + math::Transform &from, + math::Transform &to) { + return std::vector<entity::EntityID>(); + } + + InternalPhysicsEngine::InternalPhysicsEngine(entity::EntityManager *entities) : m_entities(entities) { + + } + +} \ No newline at end of file diff --git a/demo/CMakeLists.txt b/demo/CMakeLists.txt index 665ed51..1530839 100644 --- a/demo/CMakeLists.txt +++ b/demo/CMakeLists.txt @@ -23,7 +23,7 @@ target_include_directories(demo ${CMAKE_CURRENT_SOURCE_DIR}/include ) -target_link_libraries( demo fggl ) +target_link_libraries( demo fggl fggl-phys ) #target_link_libraries(demo fggl fgglbt) if ( FGGL_EXT_LUA ) target_link_libraries( demo fggl-lua ) diff --git a/demo/demo/grid.cpp b/demo/demo/grid.cpp index 3af2469..51e7109 100644 --- a/demo/demo/grid.cpp +++ b/demo/demo/grid.cpp @@ -109,7 +109,7 @@ namespace demo { std::function<void(void)> callback; }; - GridScene::GridScene(fggl::App &app) : GameBase(app), m_tiles(), m_animator(15.0F), m_grid(nullptr), m_canvas() { + GridScene::GridScene(fggl::App &app) : GameBase(app), m_tiles(), m_grid(nullptr), m_canvas(), m_animator(15.0F) { m_animator.add([this](){this->tickPlayer();}); auto btnGrid = std::make_unique<fggl::gui::GridBox>(0, 2); diff --git a/demo/demo/main.cpp b/demo/demo/main.cpp index 7960f63..d06c75f 100644 --- a/demo/demo/main.cpp +++ b/demo/demo/main.cpp @@ -24,7 +24,7 @@ #if __has_include("fggl/phys/bullet/bullet.hpp") #include "fggl/phys/bullet/bullet.hpp" #else - #include "fggl/phys/null.hpp" + #include <phys/module.hpp> #endif #include "fggl/fggl.hpp" @@ -100,7 +100,7 @@ int main(int argc, const char* argv[]) { #ifdef FGGL_MODULE_BULLET moduleManager.use<fggl::phys::Bullet3>(); #else - moduleManager.use<fggl::phys::NullPhysics>(); + moduleManager.use<fggl::phys::InternalPhysics>(); #endif moduleManager.resolve(); diff --git a/include/fggl/entity/gridworld/zone.hpp b/include/fggl/entity/gridworld/zone.hpp index 28d0c05..66953b0 100644 --- a/include/fggl/entity/gridworld/zone.hpp +++ b/include/fggl/entity/gridworld/zone.hpp @@ -104,8 +104,8 @@ namespace fggl::entity::grid { [[nodiscard]] inline bool inBounds(GridPos pos) const { - return 0 <= pos.x && pos.x <= width - && 0 <= pos.y && pos.y <= height; + return 0U <= pos.x && pos.x <= width + && 0U <= pos.y && pos.y <= height; } void clear() { diff --git a/include/fggl/grid/hexagon.hpp b/include/fggl/grid/hexagon.hpp index 7277d65..3573a10 100644 --- a/include/fggl/grid/hexagon.hpp +++ b/include/fggl/grid/hexagon.hpp @@ -25,6 +25,7 @@ #include <compare> #include <fggl/math/fmath.hpp> +#include <fggl/math/easing.hpp> /** * Hexagonal Grid Implementation. diff --git a/include/fggl/math/easing.hpp b/include/fggl/math/easing.hpp index f31eb90..2c07c2e 100644 --- a/include/fggl/math/easing.hpp +++ b/include/fggl/math/easing.hpp @@ -24,8 +24,12 @@ namespace fggl::math { - constexpr inline float lerp(float a, float b, float w) { - return (b - a) * w + a; + constexpr float lerpImprecise(float start, float end, float t) { + return start + t * (end - start); + } + + constexpr float lerp(float start, float end, float t) { + return (1 - t) * start + t * end; } constexpr inline float scale(float in, float inMin, float inMax, float outMin, float outMax) { diff --git a/include/fggl/math/fmath.hpp b/include/fggl/math/fmath.hpp index d708837..c2f81de 100644 --- a/include/fggl/math/fmath.hpp +++ b/include/fggl/math/fmath.hpp @@ -29,6 +29,11 @@ namespace fggl::math { * A 3D floating-point vector. */ using vec3f = glm::vec3; + constexpr static const math::vec3f VEC3_ZERO{0.0F, 0.0F, 0.0F}; + + constexpr vec3f add_scaled(vec3f vec, vec3f value, float scaled) { + return vec + (value * scaled); + } /** * A 2D floating-point vector. @@ -36,7 +41,6 @@ namespace fggl::math { using vec2f = glm::vec2; constexpr static const math::vec2f VEC2_ZERO{0.0F, 0.0F}; - constexpr static const math::vec3f VEC3_ZERO{0.0F, 0.0F, 0.0F}; constexpr static const math::vec3f VEC3_ONES{1.0F, 1.0F, 1.0F}; /** @@ -84,14 +88,6 @@ namespace fggl::math { return valueOrMin > max ? max : valueOrMin; } - constexpr float lerpImprecise(float start, float end, float t) { - return start + t * (end - start); - } - - constexpr float lerp(float start, float end, float t) { - return (1 - t) * start + t * end; - } - template<typename T> [[nodiscard]] constexpr T smooth_add(T first, T second, const float weight) { -- GitLab