From e5349cc093f65951f6096c913b837049f6e04112 Mon Sep 17 00:00:00 2001 From: Dominic Masters Date: Wed, 11 Sep 2024 09:21:56 -0500 Subject: [PATCH] Created basic movement. --- src/dawn/util/Easing.cpp | 51 +++++++++++++ src/dawn/util/Easing.hpp | 96 +++++++++++++++++++++++++ src/dawnglfw/input/InputManager.cpp | 48 ++++++++----- src/dawnrpg/component/entity/Entity.cpp | 65 +++++++++++++---- src/dawnrpg/component/entity/Entity.hpp | 30 +++++++- src/dawnrpg/component/entity/Player.cpp | 38 +++++++--- src/dawnrpg/component/entity/Player.hpp | 4 ++ src/dawnrpg/input/InputBinds.hpp | 3 +- src/dawnrpg/prefab/CMakeLists.txt | 1 + src/dawnrpg/prefab/PlayerPrefab.cpp | 10 +-- src/dawnrpg/prefab/TestEntityPrefab.cpp | 37 ++++++++++ src/dawnrpg/prefab/TestEntityPrefab.hpp | 24 +++++++ src/dawnrpg/scenes/WorldScene.cpp | 5 ++ 13 files changed, 364 insertions(+), 48 deletions(-) create mode 100644 src/dawnrpg/prefab/TestEntityPrefab.cpp create mode 100644 src/dawnrpg/prefab/TestEntityPrefab.hpp diff --git a/src/dawn/util/Easing.cpp b/src/dawn/util/Easing.cpp index 095f31a2..ad48d16a 100644 --- a/src/dawn/util/Easing.cpp +++ b/src/dawn/util/Easing.cpp @@ -45,4 +45,55 @@ float_t Easing::easeOutQuart(float_t t) { float_t Easing::easeInOutQuart(float_t t) { return t < 0.5 ? 8 * t * t * t * t : 1 - 8 * (--t) * t * t * t; +} + +float_t Easing::easeInQuint(float_t t) { + return t * t * t * t * t; +} + +float_t Easing::easeOutQuint(float_t t) { + return 1 + (--t) * t * t * t * t; +} + +float_t Easing::easeInOutQuint(float_t t) { + return t < 0.5 ? 16 * t * t * t * t * t : 1 + 16 * (--t) * t * t * t * t; +} + +float_t Easing::easeInSine(float_t t) { + return 1 - cos(t * M_PI_2); +} + +float_t Easing::easeOutSine(float_t t) { + return sin(t * M_PI_2); +} + +float_t Easing::easeInOutSine(float_t t) { + return 0.5 * (1 - cos(M_PI * t)); +} + +float_t Easing::easeInExpo(float_t t) { + return t == 0 ? 0 : pow(2, 10 * (t - 1)); +} + +float_t Easing::easeOutExpo(float_t t) { + return t == 1 ? 1 : 1 - pow(2, -10 * t); +} + +float_t Easing::easeInOutExpo(float_t t) { + if(t == 0) return 0; + if(t == 1) return 1; + if(t < 0.5) return 0.5 * pow(2, 20 * t - 10); + return 1 - 0.5 * pow(2, -20 * t + 10); +} + +float_t Easing::easeInCirc(float_t t) { + return 1 - sqrt(1 - t * t); +} + +float_t Easing::easeOutCirc(float_t t) { + return sqrt(1 - (--t) * t); +} + +float_t Easing::easeInOutCirc(float_t t) { + return t < 0.5 ? 0.5 * (1 - sqrt(1 - 4 * t * t)) : 0.5 * (sqrt(-((2 * t) - 3) * ((2 * t) - 1)) + 1); } \ No newline at end of file diff --git a/src/dawn/util/Easing.hpp b/src/dawn/util/Easing.hpp index bece5145..d22b8668 100644 --- a/src/dawn/util/Easing.hpp +++ b/src/dawn/util/Easing.hpp @@ -88,5 +88,101 @@ namespace Dawn { * @return Eased value. */ static float_t easeInOutQuart(float_t t); + + /** + * Quintic easing function. + * + * @param t Time value between 0 and 1. + * @return Eased value. + */ + static float_t easeInQuint(float_t t); + + /** + * Quintic easing function. + * + * @param t Time value between 0 and 1. + * @return Eased value. + */ + static float_t easeOutQuint(float_t t); + + /** + * Quintic easing function. + * + * @param t Time value between 0 and 1. + * @return Eased value. + */ + static float_t easeInOutQuint(float_t t); + + /** + * Sine easing function. + * + * @param t Time value between 0 and 1. + * @return Eased value. + */ + static float_t easeInSine(float_t t); + + /** + * Sine easing function. + * + * @param t Time value between 0 and 1. + * @return Eased value. + */ + static float_t easeOutSine(float_t t); + + /** + * Sine easing function. + * + * @param t Time value between 0 and 1. + * @return Eased value. + */ + static float_t easeInOutSine(float_t t); + + /** + * Exponential easing function. + * + * @param t Time value between 0 and 1. + * @return Eased value. + */ + static float_t easeInExpo(float_t t); + + /** + * Exponential easing function. + * + * @param t Time value between 0 and 1. + * @return Eased value. + */ + static float_t easeOutExpo(float_t t); + + /** + * Exponential easing function. + * + * @param t Time value between 0 and 1. + * @return Eased value. + */ + static float_t easeInOutExpo(float_t t); + + /** + * Circular easing function. + * + * @param t Time value between 0 and 1. + * @return Eased value. + */ + static float_t easeInCirc(float_t t); + + /** + * Circular easing function. + * + * @param t Time value between 0 and 1. + * @return Eased value. + */ + static float_t easeOutCirc(float_t t); + + /** + * Circular easing function. + * + * @param t Time value between 0 and 1. + * @return Eased value. + */ + static float_t easeInOutCirc(float_t t); }; } \ No newline at end of file diff --git a/src/dawnglfw/input/InputManager.cpp b/src/dawnglfw/input/InputManager.cpp index 5df75335..588bd139 100644 --- a/src/dawnglfw/input/InputManager.cpp +++ b/src/dawnglfw/input/InputManager.cpp @@ -17,6 +17,7 @@ void InputManager::init(const std::shared_ptr game) { double_t y ) { auto game = (Game*)glfwGetWindowUserPointer(window); + assertNotNull(game, "Game is not set on window user pointer."); game->inputManager.rawInputValues[INPUT_MANAGER_AXIS_MOUSE_X] = (float_t) x; game->inputManager.rawInputValues[INPUT_MANAGER_AXIS_MOUSE_Y] = (float_t) y; }); @@ -28,21 +29,17 @@ void InputManager::init(const std::shared_ptr game) { int32_t mods ) { auto game = (Game*)glfwGetWindowUserPointer(window); - game->inputManager.rawInputValues[INPUT_MANAGER_AXIS_MOUSE_0 + button] = ( - action == GLFW_PRESS ? 1.0f : 0.0f - ); - }); - - glfwSetMouseButtonCallback(window, []( - GLFWwindow* window, - int32_t button, - int32_t action, - int32_t mods - ) { - auto game = (Game*)glfwGetWindowUserPointer(window); - game->inputManager.rawInputValues[button] = ( - action == GLFW_PRESS ? 1.0f : 0.0f - ); + assertNotNull(game, "Game is not set on window user pointer."); + switch(action) { + case GLFW_PRESS: + game->inputManager.rawInputValues[INPUT_MANAGER_AXIS_MOUSE_0 + button] = 1.0f; + return; + case GLFW_RELEASE: + game->inputManager.rawInputValues[INPUT_MANAGER_AXIS_MOUSE_0 + button] = 0.0f; + return; + default: + return; + } }); glfwSetKeyCallback(window, []( @@ -53,9 +50,18 @@ void InputManager::init(const std::shared_ptr game) { int32_t mods ) { auto game = (Game*)glfwGetWindowUserPointer(window); - game->inputManager.rawInputValues[key] = ( - action == GLFW_PRESS ? 1.0f : 0.0f - ); + assertNotNull(game, "Game is not set on window user pointer."); + + switch(action) { + case GLFW_PRESS: + game->inputManager.rawInputValues[key] = 1.0f; + return; + case GLFW_RELEASE: + game->inputManager.rawInputValues[key] = 0.0f; + return; + default: + return; + } }); // Set default values @@ -63,6 +69,12 @@ void InputManager::init(const std::shared_ptr game) { this->bind(InputBind::Up, GLFW_KEY_UP); this->bind(InputBind::Down, GLFW_KEY_S); this->bind(InputBind::Down, GLFW_KEY_DOWN); + this->bind(InputBind::Left, GLFW_KEY_A); + this->bind(InputBind::Left, GLFW_KEY_LEFT); + this->bind(InputBind::Right, GLFW_KEY_D); + this->bind(InputBind::Right, GLFW_KEY_RIGHT); + this->bind(InputBind::Run, GLFW_KEY_LEFT_SHIFT); + this->bind(InputBind::Run, GLFW_KEY_RIGHT_SHIFT); } float_t InputManager::getInputValue(int32_t axis) { diff --git a/src/dawnrpg/component/entity/Entity.cpp b/src/dawnrpg/component/entity/Entity.cpp index 1d38cdc0..341d4c6d 100644 --- a/src/dawnrpg/component/entity/Entity.cpp +++ b/src/dawnrpg/component/entity/Entity.cpp @@ -6,17 +6,41 @@ #include "Entity.hpp" #include "scene/Scene.hpp" #include "assert/assert.hpp" +#include "util/Easing.hpp" using namespace Dawn; +const glm::vec3 EntityTilePosition::toWorldSpace() { + return glm::vec3( + this->x, + this->y, + this->z + ); +} + void Entity::onInit() { auto world = this->world.lock(); assertNotNull(world, "World is no longer active?"); listeners.push_back(getScene()->onUnpausedUpdate.listen([this](float delta) { if(this->stepTime <= 0.0f) return; - this->stepTime = 0; - this->getItem()->setLocalPosition(glm::vec3(this->tileX, this->tileY, this->tileZ)); + this->stepTime -= delta * this->stepSpeed; + + glm::vec3 nextPosition = this->tilePosition.toWorldSpace(); + + if(this->stepTime <= 0) { + this->getItem()->setLocalPosition(nextPosition); + this->stepTime = 0.0f; + this->eventMove.emit(); + this->eventStepEnd.emit(); + return; + } + + float_t t = Easing::easeInOutSine(1.0f - this->stepTime); + glm::vec3 lastPosition = this->lastTilePosition.toWorldSpace(); + glm::vec3 newLocal = glm::mix(lastPosition, nextPosition, t); + this->getItem()->setLocalPosition(newLocal); + this->eventMove.emit(); })); } @@ -24,24 +48,24 @@ void Entity::onDispose() { } -void Entity::move(const enum EntityDirection direction) { - entitytileposition_t newTileX, newTileY, newTileZ; - newTileX = tileX; - newTileY = tileY; - newTileZ = tileZ; +void Entity::move( + const enum EntityDirection direction, + const float_t stepSpeed +) { + struct EntityTilePosition newPosition = this->tilePosition; switch(direction) { case EntityDirection::Up: - newTileZ--; + newPosition.z--; break; case EntityDirection::Down: - newTileZ++; + newPosition.z++; break; case EntityDirection::Left: - newTileX--; + newPosition.x--; break; case EntityDirection::Right: - newTileX++; + newPosition.x++; break; default: assertUnreachable("Invalid direction: %d", direction); @@ -51,9 +75,22 @@ void Entity::move(const enum EntityDirection direction) { // If the tile is not walkable, return early. // Move the entity to the new tile. - this->tileX = newTileX; - this->tileY = newTileY; - this->tileZ = newTileZ; + this->lastTilePosition = this->tilePosition; + this->tilePosition = newPosition; this->direction = direction; this->stepTime = 1.0f; + this->stepSpeed = stepSpeed; + + this->eventStepStart.emit(); +} + +void Entity::setPosition(struct EntityTilePosition pos) { + this->stepTime = 0.0f; + this->tilePosition = pos; + this->lastTilePosition = pos; + this->getItem()->setLocalPosition(pos.toWorldSpace()); +} + +bool_t Entity::isMoving() { + return this->stepTime > 0.0f; } \ No newline at end of file diff --git a/src/dawnrpg/component/entity/Entity.hpp b/src/dawnrpg/component/entity/Entity.hpp index 7d7804ef..61eb4dc6 100644 --- a/src/dawnrpg/component/entity/Entity.hpp +++ b/src/dawnrpg/component/entity/Entity.hpp @@ -5,10 +5,14 @@ #pragma once #include "scene/SceneComponent.hpp" +#include "event/Event.hpp" typedef uint32_t entityid_t; typedef int64_t entitytileposition_t; +#define ENTITY_STEP_SPEED_DEFAULT 3.0f +#define ENTITY_STEP_SPEED_RUNNING 6.0f + namespace Dawn { class World; @@ -18,20 +22,42 @@ namespace Dawn { Left, Right }; + + struct EntityTilePosition { + entitytileposition_t x, y, z; + + /** + * Converts the tile position to a world space position. + * + * @return This tile position in world space. + */ + const glm::vec3 toWorldSpace(); + }; class Entity : public SceneComponent { private: - entitytileposition_t tileX = 0, tileY = 0, tileZ = 0; + struct EntityTilePosition lastTilePosition; enum EntityDirection direction = EntityDirection::Down; float_t stepTime = 0.0f; + float_t stepSpeed = ENTITY_STEP_SPEED_DEFAULT; protected: + struct EntityTilePosition tilePosition; public: std::weak_ptr world; + Event<> eventStepStart; + Event<> eventStepEnd; + Event<> eventMove; void onInit() override; void onDispose() override; - void move(const enum EntityDirection direction); + void move( + const enum EntityDirection direction, + const float_t stepSpeed = ENTITY_STEP_SPEED_DEFAULT + ); + void setPosition(struct EntityTilePosition position); + + bool_t isMoving(); }; } \ No newline at end of file diff --git a/src/dawnrpg/component/entity/Player.cpp b/src/dawnrpg/component/entity/Player.cpp index fc8f7601..ae10a518 100644 --- a/src/dawnrpg/component/entity/Player.cpp +++ b/src/dawnrpg/component/entity/Player.cpp @@ -13,17 +13,39 @@ void Player::onInit() { Entity::onInit(); listeners.push_back(getScene()->onUnpausedUpdate.listen([this](float delta) { + if(this->isMoving()) return; + InputManager &im = getGame()->inputManager; - if(im.isPressed(InputBind::Up)) { - this->move(EntityDirection::Up); - } else if(im.isPressed(InputBind::Down)) { - this->move(EntityDirection::Down); - } else if(im.isPressed(InputBind::Left)) { - this->move(EntityDirection::Left); - } else if(im.isPressed(InputBind::Right)) { - this->move(EntityDirection::Right); + bool_t run = im.isDown(InputBind::Run); + float_t stepSpeed = run ? ENTITY_STEP_SPEED_RUNNING : ENTITY_STEP_SPEED_DEFAULT; + + if(im.isDown(InputBind::Up)) { + this->move(EntityDirection::Up, stepSpeed); + } else if(im.isDown(InputBind::Down)) { + this->move(EntityDirection::Down, stepSpeed); + } else if(im.isDown(InputBind::Left)) { + this->move(EntityDirection::Left, stepSpeed); + } else if(im.isDown(InputBind::Right)) { + this->move(EntityDirection::Right, stepSpeed); } })); + + listeners.push_back(eventMove.listen([this]() { + this->updateCameraPosition(); + })); + + this->updateCameraPosition(); +} + +void Player::updateCameraPosition() { + auto c = camera.lock(); + if(!c) return; + glm::vec3 pos = this->getItem()->getLocalPosition(); + c->getItem()->lookAt( + pos + glm::vec3(0, 8, 1), + pos, + glm::vec3(0, 1, 0) + ); } void Player::onDispose() { diff --git a/src/dawnrpg/component/entity/Player.hpp b/src/dawnrpg/component/entity/Player.hpp index b37bdf48..ff3f4de8 100644 --- a/src/dawnrpg/component/entity/Player.hpp +++ b/src/dawnrpg/component/entity/Player.hpp @@ -5,12 +5,16 @@ #pragma once #include "component/entity/Entity.hpp" +#include "component/display/Camera.hpp" namespace Dawn { class Player : public Entity { protected: + void updateCameraPosition(); public: + std::weak_ptr camera; + void onInit() override; void onDispose() override; }; diff --git a/src/dawnrpg/input/InputBinds.hpp b/src/dawnrpg/input/InputBinds.hpp index 38cea978..7f97c7b1 100644 --- a/src/dawnrpg/input/InputBinds.hpp +++ b/src/dawnrpg/input/InputBinds.hpp @@ -10,6 +10,7 @@ namespace Dawn { Up, Down, Left, - Right + Right, + Run }; } \ No newline at end of file diff --git a/src/dawnrpg/prefab/CMakeLists.txt b/src/dawnrpg/prefab/CMakeLists.txt index 109193b1..7ba151cd 100644 --- a/src/dawnrpg/prefab/CMakeLists.txt +++ b/src/dawnrpg/prefab/CMakeLists.txt @@ -7,4 +7,5 @@ target_sources(${DAWN_TARGET_NAME} PRIVATE PlayerPrefab.cpp WorldPrefab.cpp + TestEntityPrefab.cpp ) \ No newline at end of file diff --git a/src/dawnrpg/prefab/PlayerPrefab.cpp b/src/dawnrpg/prefab/PlayerPrefab.cpp index 0202a1e2..bfd5a0fd 100644 --- a/src/dawnrpg/prefab/PlayerPrefab.cpp +++ b/src/dawnrpg/prefab/PlayerPrefab.cpp @@ -4,7 +4,7 @@ // https://opensource.org/licenses/MIT #include "PlayerPrefab.hpp" -#include "display/mesh/QuadMesh.hpp" +#include "display/mesh/CubeMesh.hpp" using namespace Dawn; @@ -19,11 +19,11 @@ struct PlayerPrefab Dawn::createPlayerPrefab( player.player->world = world; player.mesh = std::make_shared(); - player.mesh->createBuffers(QUAD_VERTICE_COUNT, QUAD_INDICE_COUNT); - QuadMesh::buffer( + player.mesh->createBuffers(CUBE_VERTICE_COUNT, CUBE_INDICE_COUNT); + CubeMesh::buffer( player.mesh, - glm::vec4(0, 0, 1, 1), - glm::vec4(0, 0, 1, 1), + glm::vec3(-0.5f, -0.5f, -0.5f), + glm::vec3(1, 1, 1), 0, 0 ); diff --git a/src/dawnrpg/prefab/TestEntityPrefab.cpp b/src/dawnrpg/prefab/TestEntityPrefab.cpp new file mode 100644 index 00000000..43835bda --- /dev/null +++ b/src/dawnrpg/prefab/TestEntityPrefab.cpp @@ -0,0 +1,37 @@ +// Copyright (c) 2024 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#include "TestEntityPrefab.hpp" +#include "display/mesh/CubeMesh.hpp" + +using namespace Dawn; + +struct TestEntityPrefab Dawn::createTestEntityPrefab( + const std::shared_ptr world +) { + struct TestEntityPrefab entity; + entity.item = world->getScene()->createSceneItem(); + entity.item->setParent(world->getItem()); + + entity.entity = entity.item->addComponent(); + entity.entity->world = world; + + entity.mesh = std::make_shared(); + entity.mesh->createBuffers(CUBE_VERTICE_COUNT, CUBE_INDICE_COUNT); + CubeMesh::buffer( + entity.mesh, + glm::vec3(-0.5f, -0.5f, -0.5f), + glm::vec3(1, 1, 1), + 0, 0 + ); + + entity.meshRenderer = entity.item->addComponent(); + entity.meshRenderer->mesh = entity.mesh; + + entity.material = entity.item->addComponent(); + entity.material->setColor(COLOR_GREEN); + + return entity; +} \ No newline at end of file diff --git a/src/dawnrpg/prefab/TestEntityPrefab.hpp b/src/dawnrpg/prefab/TestEntityPrefab.hpp new file mode 100644 index 00000000..b8805719 --- /dev/null +++ b/src/dawnrpg/prefab/TestEntityPrefab.hpp @@ -0,0 +1,24 @@ +// Copyright (c) 2024 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#pragma once +#include "prefab/Prefab.hpp" +#include "component/world/World.hpp" +#include "component/display/MeshRenderer.hpp" +#include "component/display/material/SimpleTexturedMaterial.hpp" + +namespace Dawn { + struct TestEntityPrefab : public Prefab { + public: + std::shared_ptr entity; + std::shared_ptr mesh; + std::shared_ptr meshRenderer; + std::shared_ptr material; + }; + + struct TestEntityPrefab createTestEntityPrefab( + const std::shared_ptr world + ); +} \ No newline at end of file diff --git a/src/dawnrpg/scenes/WorldScene.cpp b/src/dawnrpg/scenes/WorldScene.cpp index 5f860d26..cb534af7 100644 --- a/src/dawnrpg/scenes/WorldScene.cpp +++ b/src/dawnrpg/scenes/WorldScene.cpp @@ -6,6 +6,7 @@ #include "scenes/SceneList.hpp" #include "prefab/PlayerPrefab.hpp" +#include "prefab/TestEntityPrefab.hpp" #include "prefab/WorldPrefab.hpp" #include "component/display/Camera.hpp" // #include "prefab/SimpleSpinningCube.hpp" @@ -43,4 +44,8 @@ void Dawn::worldScene(Scene &s) { auto world = createWorldPrefab(s); auto player = createPlayerPrefab(world.world); + player.player->camera = camera; + + auto test = createTestEntityPrefab(world.world); + test.entity->setPosition({ 5, 0, 0 }); } \ No newline at end of file