From ad3974bde59dc9efac995ae89ca5fc9fa4fc383a Mon Sep 17 00:00:00 2001 From: Dominic Masters Date: Wed, 11 Sep 2024 10:04:54 -0500 Subject: [PATCH] Testing walking. --- src/dawn/assert/assert.hpp | 11 +++ src/dawnrpg/component/entity/Entity.cpp | 41 +++++++++-- src/dawnrpg/component/entity/Entity.hpp | 71 +++++++++++++++---- src/dawnrpg/component/entity/EntityID.hpp | 15 ++++ .../component/entity/EntityTilePosition.cpp | 16 +++++ .../component/entity/EntityTilePosition.hpp | 30 ++++++++ src/dawnrpg/component/entity/Player.cpp | 2 +- src/dawnrpg/component/world/World.cpp | 43 +++++++++++ src/dawnrpg/component/world/World.hpp | 35 ++++++++- src/dawnrpg/prefab/PlayerPrefab.cpp | 1 + src/dawnrpg/prefab/TestEntityPrefab.cpp | 1 + 11 files changed, 243 insertions(+), 23 deletions(-) create mode 100644 src/dawnrpg/component/entity/EntityID.hpp create mode 100644 src/dawnrpg/component/entity/EntityTilePosition.cpp create mode 100644 src/dawnrpg/component/entity/EntityTilePosition.hpp diff --git a/src/dawn/assert/assert.hpp b/src/dawn/assert/assert.hpp index 223b4b06..b33bf94c 100644 --- a/src/dawn/assert/assert.hpp +++ b/src/dawn/assert/assert.hpp @@ -84,6 +84,17 @@ void assertTrueImplement( map.find(key) != map.end(), __VA_ARGS__ \ ) +/** + * Asserts that a given map does not have a specific key. + * @param map Map to check. + * @param key Key to check for. + * @param message Message (sprintf format) to send to the logger. + * @param args Optional TParam args for the sprintf message to accept. + */ +#define assertMapNotHasKey(map, key, ...) assertTrue( \ + map.find(key) == map.end(), __VA_ARGS__ \ +) + /** * Asserts that a given value has a specific flag turned off. * diff --git a/src/dawnrpg/component/entity/Entity.cpp b/src/dawnrpg/component/entity/Entity.cpp index 341d4c6d..fa87a8ee 100644 --- a/src/dawnrpg/component/entity/Entity.cpp +++ b/src/dawnrpg/component/entity/Entity.cpp @@ -7,6 +7,7 @@ #include "scene/Scene.hpp" #include "assert/assert.hpp" #include "util/Easing.hpp" +#include "component/world/World.hpp" using namespace Dawn; @@ -19,10 +20,13 @@ const glm::vec3 EntityTilePosition::toWorldSpace() { } void Entity::onInit() { - auto world = this->world.lock(); - assertNotNull(world, "World is no longer active?"); + listeners.push_back(getScene()->onUnpausedUpdate.listen([this](float_t delta){ + if(this->turnTime <= 0.0f) return; + this->turnTime -= delta; + if(this->turnTime <= 0) this->turnTime = 0.0f; + })); - listeners.push_back(getScene()->onUnpausedUpdate.listen([this](float delta) { + listeners.push_back(getScene()->onUnpausedUpdate.listen([this](float_t delta){ if(this->stepTime <= 0.0f) return; this->stepTime -= delta * this->stepSpeed; @@ -42,16 +46,26 @@ void Entity::onInit() { this->getItem()->setLocalPosition(newLocal); this->eventMove.emit(); })); + + this->getWorld()->entityNotifyInit(this->getItem()); } void Entity::onDispose() { - + this->getWorld()->entityNotifyDispose(this->getItem()); } -void Entity::move( +std::shared_ptr Entity::getWorld() { + auto world = this->world.lock(); + assertNotNull(world, "World is no longer active?"); + return world; +} + +struct EntityStepResult Entity::move( const enum EntityDirection direction, const float_t stepSpeed ) { + assertFalse(this->isMoving(), "Entity is already moving."); + struct EntityStepResult result; struct EntityTilePosition newPosition = this->tilePosition; switch(direction) { @@ -71,6 +85,15 @@ void Entity::move( assertUnreachable("Invalid direction: %d", direction); } + // Check for entity in way. + auto entityInWay = this->getWorld()->getEntityAt(newPosition); + if(entityInWay) { + result.type = EntityStepResultType::EntityInWay; + result.entityInWay = entityInWay; + this->turn(direction); + return result; + } + // Get the tile at the destination, check for height if we are on stairs, etc. // If the tile is not walkable, return early. @@ -82,6 +105,11 @@ void Entity::move( this->stepSpeed = stepSpeed; this->eventStepStart.emit(); + return result; +} + +void Entity::turn(const enum EntityDirection direction) { + this->direction = direction; } void Entity::setPosition(struct EntityTilePosition pos) { @@ -89,8 +117,9 @@ void Entity::setPosition(struct EntityTilePosition pos) { this->tilePosition = pos; this->lastTilePosition = pos; this->getItem()->setLocalPosition(pos.toWorldSpace()); + this->eventMove.emit(); } bool_t Entity::isMoving() { - return this->stepTime > 0.0f; + return this->stepTime > 0.0f || this->turnTime > 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 61eb4dc6..3d79430b 100644 --- a/src/dawnrpg/component/entity/Entity.hpp +++ b/src/dawnrpg/component/entity/Entity.hpp @@ -6,15 +6,15 @@ #pragma once #include "scene/SceneComponent.hpp" #include "event/Event.hpp" - -typedef uint32_t entityid_t; -typedef int64_t entitytileposition_t; +#include "EntityID.hpp" +#include "EntityTilePosition.hpp" #define ENTITY_STEP_SPEED_DEFAULT 3.0f #define ENTITY_STEP_SPEED_RUNNING 6.0f namespace Dawn { class World; + class Entity; enum class EntityDirection { Up, @@ -22,22 +22,25 @@ 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(); + + enum class EntityStepResultType { + Step, + EntityInWay, + }; + + struct EntityStepResult { + enum EntityStepResultType type; + + // I'd love to unionize this but it seems that it's not ideal rn. + std::shared_ptr entityInWay; }; class Entity : public SceneComponent { private: - struct EntityTilePosition lastTilePosition; enum EntityDirection direction = EntityDirection::Down; + float_t turnTime = 0.0f; + + struct EntityTilePosition lastTilePosition; float_t stepTime = 0.0f; float_t stepSpeed = ENTITY_STEP_SPEED_DEFAULT; @@ -45,19 +48,57 @@ namespace Dawn { struct EntityTilePosition tilePosition; public: + enum EntityID id = EntityID::Null; std::weak_ptr world; + Event<> eventStepStart; Event<> eventStepEnd; Event<> eventMove; void onInit() override; void onDispose() override; - void move( + + /** + * Gets the world this entity is in. + * + * @return The world this entity is in. + */ + std::shared_ptr getWorld(); + + /** + * Moves the entity in the given direction. + * + * @param direction The direction to move in. + * @param stepSpeed The speed to move at. + */ + struct EntityStepResult move( const enum EntityDirection direction, const float_t stepSpeed = ENTITY_STEP_SPEED_DEFAULT ); + + /** + * Turns the entity in the given direction. This will have a slight delay + * and count as moving. + * + * @param direction The direction to turn in. + */ + void turn(const enum EntityDirection direction); + + /** + * Safely sets the position of the entity. This will invoke movement + * events but not step events, so steps may be cancelled in this context. + * + * @param position The new position of the entity. + */ void setPosition(struct EntityTilePosition position); + /** + * Checks if the entity is currently moving (in a step). + * + * @return True if the entity is moving, false otherwise. + */ bool_t isMoving(); + + friend class World; }; } \ No newline at end of file diff --git a/src/dawnrpg/component/entity/EntityID.hpp b/src/dawnrpg/component/entity/EntityID.hpp new file mode 100644 index 00000000..26122e3d --- /dev/null +++ b/src/dawnrpg/component/entity/EntityID.hpp @@ -0,0 +1,15 @@ +// Copyright (c) 2024 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#pragma once +#include "Entity.hpp" + +namespace Dawn { + enum class EntityID : uint32_t { + Null = 0, + Player = 1, + TestEntity = 2 + }; +} \ No newline at end of file diff --git a/src/dawnrpg/component/entity/EntityTilePosition.cpp b/src/dawnrpg/component/entity/EntityTilePosition.cpp new file mode 100644 index 00000000..6b4ffc75 --- /dev/null +++ b/src/dawnrpg/component/entity/EntityTilePosition.cpp @@ -0,0 +1,16 @@ +// Copyright (c) 2024 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#include "EntityTilePosition.hpp" + +using namespace Dawn; + +const glm::vec3 EntityTilePosition::toWorldSpace() { + return glm::vec3( + this->x, + this->y, + this->z + ); +} \ No newline at end of file diff --git a/src/dawnrpg/component/entity/EntityTilePosition.hpp b/src/dawnrpg/component/entity/EntityTilePosition.hpp new file mode 100644 index 00000000..56ae432b --- /dev/null +++ b/src/dawnrpg/component/entity/EntityTilePosition.hpp @@ -0,0 +1,30 @@ +// Copyright (c) 2024 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#pragma once +#include "dawnlibs.hpp" + +namespace Dawn { + struct EntityTilePosition { + int64_t x, y, z; + + // Overload the equality operator. + bool operator==(const EntityTilePosition &other) const { + return this->x == other.x && this->y == other.y && this->z == other.z; + } + + // Overload the inequality operator. + bool operator!=(const EntityTilePosition &other) const { + return !(*this == other); + } + + /** + * Converts the tile position to a world space position. + * + * @return This tile position in world space. + */ + const glm::vec3 toWorldSpace(); + }; +} \ No newline at end of file diff --git a/src/dawnrpg/component/entity/Player.cpp b/src/dawnrpg/component/entity/Player.cpp index ae10a518..615a92e8 100644 --- a/src/dawnrpg/component/entity/Player.cpp +++ b/src/dawnrpg/component/entity/Player.cpp @@ -12,7 +12,7 @@ using namespace Dawn; void Player::onInit() { Entity::onInit(); - listeners.push_back(getScene()->onUnpausedUpdate.listen([this](float delta) { + listeners.push_back(getScene()->onUnpausedUpdate.listen([this](float_t delta){ if(this->isMoving()) return; InputManager &im = getGame()->inputManager; diff --git a/src/dawnrpg/component/world/World.cpp b/src/dawnrpg/component/world/World.cpp index e8c4b07f..c5a61e4f 100644 --- a/src/dawnrpg/component/world/World.cpp +++ b/src/dawnrpg/component/world/World.cpp @@ -4,11 +4,54 @@ // https://opensource.org/licenses/MIT #include "World.hpp" +#include "scene/Scene.hpp" using namespace Dawn; +void World::entityNotifyInit(std::shared_ptr item) { + auto entity = item->getComponent(); + assertNotNull(entity, "Entity component not found on item."); + auto id = entity->id; + assertMapNotHasKey(this->entities, id, "Entity already exists in world."); + assertTrue(id != EntityID::Null, "Entity ID is invalid."); + this->entities[id] = entity; +} + +void World::entityNotifyDispose(std::shared_ptr item) { + auto entity = item->getComponent(); + assertNotNull(entity, "Entity component not found on item."); + auto id = entity->id; + assertMapHasKey(this->entities, id, "Entity does not exist in world."); + assertTrue(id != EntityID::Null, "Entity ID is invalid."); + this->entities.erase(id); +} + void World::onInit() { + } void World::onDispose() { + +} + +std::shared_ptr World::getEntity(const EntityID id) { + assertMapHasKey(this->entities, id, "Entity does not exist in world."); + auto ent = this->entities[id]; + auto lock = ent.lock(); + if(!lock) { + this->entities.erase(id); + return nullptr; + } + assertTrue(lock->id == id, "Entity ID mismatch."); + return lock; +} + +std::shared_ptr World::getEntityAt(const EntityTilePosition &pos) { + for(auto &pair : this->entities) { + auto ent = pair.second.lock(); + if(!ent) continue; + if(ent->tilePosition != pos) continue; + return ent; + } + return nullptr; } \ No newline at end of file diff --git a/src/dawnrpg/component/world/World.hpp b/src/dawnrpg/component/world/World.hpp index 5c366ee1..0e064c89 100644 --- a/src/dawnrpg/component/world/World.hpp +++ b/src/dawnrpg/component/world/World.hpp @@ -5,14 +5,47 @@ #pragma once #include "component/entity/Entity.hpp" +#include "component/entity/Player.hpp" namespace Dawn { class World : public SceneComponent { private: - std::map> entities; + std::unordered_map> entities; + + /** + * Part of the ECS, this function is called when an entity is initialized. + * + * @param item Scene Item that has been initialized. + */ + void entityNotifyInit(std::shared_ptr item); + + /** + * Part of the ECS, this function is called when an entity is disposed. + * + * @param item Scene Item that has been disposed. + */ + void entityNotifyDispose(std::shared_ptr entity); public: void onInit() override; void onDispose() override; + + /** + * Gets the entity with the given ID. + * + * @param id The ID of the entity to get. + * @return The entity with the given ID. + */ + std::shared_ptr getEntity(const EntityID id); + + /** + * Gets the entity at the given position. + * + * @param pos The position to check for an entity. + * @return The entity at the given position, or nullptr. + */ + std::shared_ptr getEntityAt(const EntityTilePosition &pos); + + friend class Entity; }; } \ No newline at end of file diff --git a/src/dawnrpg/prefab/PlayerPrefab.cpp b/src/dawnrpg/prefab/PlayerPrefab.cpp index bfd5a0fd..344a9165 100644 --- a/src/dawnrpg/prefab/PlayerPrefab.cpp +++ b/src/dawnrpg/prefab/PlayerPrefab.cpp @@ -17,6 +17,7 @@ struct PlayerPrefab Dawn::createPlayerPrefab( player.player = player.item->addComponent(); player.player->world = world; + player.player->id = EntityID::Player; player.mesh = std::make_shared(); player.mesh->createBuffers(CUBE_VERTICE_COUNT, CUBE_INDICE_COUNT); diff --git a/src/dawnrpg/prefab/TestEntityPrefab.cpp b/src/dawnrpg/prefab/TestEntityPrefab.cpp index 43835bda..79c81090 100644 --- a/src/dawnrpg/prefab/TestEntityPrefab.cpp +++ b/src/dawnrpg/prefab/TestEntityPrefab.cpp @@ -17,6 +17,7 @@ struct TestEntityPrefab Dawn::createTestEntityPrefab( entity.entity = entity.item->addComponent(); entity.entity->world = world; + entity.entity->id = EntityID::TestEntity; entity.mesh = std::make_shared(); entity.mesh->createBuffers(CUBE_VERTICE_COUNT, CUBE_INDICE_COUNT);