Testing walking.

This commit is contained in:
2024-09-11 10:04:54 -05:00
parent e5349cc093
commit ad3974bde5
11 changed files with 243 additions and 23 deletions

View File

@ -84,6 +84,17 @@ void assertTrueImplement(
map.find(key) != map.end(), __VA_ARGS__ \ 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. * Asserts that a given value has a specific flag turned off.
* *

View File

@ -7,6 +7,7 @@
#include "scene/Scene.hpp" #include "scene/Scene.hpp"
#include "assert/assert.hpp" #include "assert/assert.hpp"
#include "util/Easing.hpp" #include "util/Easing.hpp"
#include "component/world/World.hpp"
using namespace Dawn; using namespace Dawn;
@ -19,10 +20,13 @@ const glm::vec3 EntityTilePosition::toWorldSpace() {
} }
void Entity::onInit() { void Entity::onInit() {
auto world = this->world.lock(); listeners.push_back(getScene()->onUnpausedUpdate.listen([this](float_t delta){
assertNotNull(world, "World is no longer active?"); 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; if(this->stepTime <= 0.0f) return;
this->stepTime -= delta * this->stepSpeed; this->stepTime -= delta * this->stepSpeed;
@ -42,16 +46,26 @@ void Entity::onInit() {
this->getItem()->setLocalPosition(newLocal); this->getItem()->setLocalPosition(newLocal);
this->eventMove.emit(); this->eventMove.emit();
})); }));
this->getWorld()->entityNotifyInit(this->getItem());
} }
void Entity::onDispose() { void Entity::onDispose() {
this->getWorld()->entityNotifyDispose(this->getItem());
} }
void Entity::move( std::shared_ptr<World> 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 enum EntityDirection direction,
const float_t stepSpeed const float_t stepSpeed
) { ) {
assertFalse(this->isMoving(), "Entity is already moving.");
struct EntityStepResult result;
struct EntityTilePosition newPosition = this->tilePosition; struct EntityTilePosition newPosition = this->tilePosition;
switch(direction) { switch(direction) {
@ -71,6 +85,15 @@ void Entity::move(
assertUnreachable("Invalid direction: %d", direction); 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. // Get the tile at the destination, check for height if we are on stairs, etc.
// If the tile is not walkable, return early. // If the tile is not walkable, return early.
@ -82,6 +105,11 @@ void Entity::move(
this->stepSpeed = stepSpeed; this->stepSpeed = stepSpeed;
this->eventStepStart.emit(); this->eventStepStart.emit();
return result;
}
void Entity::turn(const enum EntityDirection direction) {
this->direction = direction;
} }
void Entity::setPosition(struct EntityTilePosition pos) { void Entity::setPosition(struct EntityTilePosition pos) {
@ -89,8 +117,9 @@ void Entity::setPosition(struct EntityTilePosition pos) {
this->tilePosition = pos; this->tilePosition = pos;
this->lastTilePosition = pos; this->lastTilePosition = pos;
this->getItem()->setLocalPosition(pos.toWorldSpace()); this->getItem()->setLocalPosition(pos.toWorldSpace());
this->eventMove.emit();
} }
bool_t Entity::isMoving() { bool_t Entity::isMoving() {
return this->stepTime > 0.0f; return this->stepTime > 0.0f || this->turnTime > 0.0f;
} }

View File

@ -6,15 +6,15 @@
#pragma once #pragma once
#include "scene/SceneComponent.hpp" #include "scene/SceneComponent.hpp"
#include "event/Event.hpp" #include "event/Event.hpp"
#include "EntityID.hpp"
typedef uint32_t entityid_t; #include "EntityTilePosition.hpp"
typedef int64_t entitytileposition_t;
#define ENTITY_STEP_SPEED_DEFAULT 3.0f #define ENTITY_STEP_SPEED_DEFAULT 3.0f
#define ENTITY_STEP_SPEED_RUNNING 6.0f #define ENTITY_STEP_SPEED_RUNNING 6.0f
namespace Dawn { namespace Dawn {
class World; class World;
class Entity;
enum class EntityDirection { enum class EntityDirection {
Up, Up,
@ -22,22 +22,25 @@ namespace Dawn {
Left, Left,
Right Right
}; };
struct EntityTilePosition { enum class EntityStepResultType {
entitytileposition_t x, y, z; Step,
EntityInWay,
/** };
* Converts the tile position to a world space position.
* struct EntityStepResult {
* @return This tile position in world space. enum EntityStepResultType type;
*/
const glm::vec3 toWorldSpace(); // I'd love to unionize this but it seems that it's not ideal rn.
std::shared_ptr<Entity> entityInWay;
}; };
class Entity : public SceneComponent { class Entity : public SceneComponent {
private: private:
struct EntityTilePosition lastTilePosition;
enum EntityDirection direction = EntityDirection::Down; enum EntityDirection direction = EntityDirection::Down;
float_t turnTime = 0.0f;
struct EntityTilePosition lastTilePosition;
float_t stepTime = 0.0f; float_t stepTime = 0.0f;
float_t stepSpeed = ENTITY_STEP_SPEED_DEFAULT; float_t stepSpeed = ENTITY_STEP_SPEED_DEFAULT;
@ -45,19 +48,57 @@ namespace Dawn {
struct EntityTilePosition tilePosition; struct EntityTilePosition tilePosition;
public: public:
enum EntityID id = EntityID::Null;
std::weak_ptr<World> world; std::weak_ptr<World> world;
Event<> eventStepStart; Event<> eventStepStart;
Event<> eventStepEnd; Event<> eventStepEnd;
Event<> eventMove; Event<> eventMove;
void onInit() override; void onInit() override;
void onDispose() override; void onDispose() override;
void move(
/**
* Gets the world this entity is in.
*
* @return The world this entity is in.
*/
std::shared_ptr<World> 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 enum EntityDirection direction,
const float_t stepSpeed = ENTITY_STEP_SPEED_DEFAULT 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); 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(); bool_t isMoving();
friend class World;
}; };
} }

View File

@ -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
};
}

View File

@ -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
);
}

View File

@ -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();
};
}

View File

@ -12,7 +12,7 @@ using namespace Dawn;
void Player::onInit() { void Player::onInit() {
Entity::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; if(this->isMoving()) return;
InputManager &im = getGame()->inputManager; InputManager &im = getGame()->inputManager;

View File

@ -4,11 +4,54 @@
// https://opensource.org/licenses/MIT // https://opensource.org/licenses/MIT
#include "World.hpp" #include "World.hpp"
#include "scene/Scene.hpp"
using namespace Dawn; using namespace Dawn;
void World::entityNotifyInit(std::shared_ptr<SceneItem> item) {
auto entity = item->getComponent<Entity>();
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<SceneItem> item) {
auto entity = item->getComponent<Entity>();
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::onInit() {
} }
void World::onDispose() { void World::onDispose() {
}
std::shared_ptr<Entity> 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<Entity> 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;
} }

View File

@ -5,14 +5,47 @@
#pragma once #pragma once
#include "component/entity/Entity.hpp" #include "component/entity/Entity.hpp"
#include "component/entity/Player.hpp"
namespace Dawn { namespace Dawn {
class World : public SceneComponent { class World : public SceneComponent {
private: private:
std::map<entityid_t, std::shared_ptr<Entity>> entities; std::unordered_map<enum EntityID, std::weak_ptr<Entity>> 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<SceneItem> 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<SceneItem> entity);
public: public:
void onInit() override; void onInit() override;
void onDispose() 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<Entity> 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<Entity> getEntityAt(const EntityTilePosition &pos);
friend class Entity;
}; };
} }

View File

@ -17,6 +17,7 @@ struct PlayerPrefab Dawn::createPlayerPrefab(
player.player = player.item->addComponent<Player>(); player.player = player.item->addComponent<Player>();
player.player->world = world; player.player->world = world;
player.player->id = EntityID::Player;
player.mesh = std::make_shared<Mesh>(); player.mesh = std::make_shared<Mesh>();
player.mesh->createBuffers(CUBE_VERTICE_COUNT, CUBE_INDICE_COUNT); player.mesh->createBuffers(CUBE_VERTICE_COUNT, CUBE_INDICE_COUNT);

View File

@ -17,6 +17,7 @@ struct TestEntityPrefab Dawn::createTestEntityPrefab(
entity.entity = entity.item->addComponent<Entity>(); entity.entity = entity.item->addComponent<Entity>();
entity.entity->world = world; entity.entity->world = world;
entity.entity->id = EntityID::TestEntity;
entity.mesh = std::make_shared<Mesh>(); entity.mesh = std::make_shared<Mesh>();
entity.mesh->createBuffers(CUBE_VERTICE_COUNT, CUBE_INDICE_COUNT); entity.mesh->createBuffers(CUBE_VERTICE_COUNT, CUBE_INDICE_COUNT);