Testing walking.
This commit is contained in:
@ -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.
|
||||||
*
|
*
|
||||||
|
@ -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;
|
||||||
}
|
}
|
@ -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;
|
||||||
};
|
};
|
||||||
}
|
}
|
15
src/dawnrpg/component/entity/EntityID.hpp
Normal file
15
src/dawnrpg/component/entity/EntityID.hpp
Normal 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
|
||||||
|
};
|
||||||
|
}
|
16
src/dawnrpg/component/entity/EntityTilePosition.cpp
Normal file
16
src/dawnrpg/component/entity/EntityTilePosition.cpp
Normal 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
|
||||||
|
);
|
||||||
|
}
|
30
src/dawnrpg/component/entity/EntityTilePosition.hpp
Normal file
30
src/dawnrpg/component/entity/EntityTilePosition.hpp
Normal 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();
|
||||||
|
};
|
||||||
|
}
|
@ -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;
|
||||||
|
@ -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;
|
||||||
}
|
}
|
@ -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;
|
||||||
};
|
};
|
||||||
}
|
}
|
@ -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);
|
||||||
|
@ -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);
|
||||||
|
Reference in New Issue
Block a user