Entity progress

This commit is contained in:
2023-03-30 21:43:13 -07:00
parent 48cf9d1584
commit ec075f0035
16 changed files with 192 additions and 53 deletions

View File

@ -21,4 +21,5 @@ add_subdirectory(scene)
set(ROSE_ASSETS_DIR ${DAWN_ASSETS_DIR}/games/rose) set(ROSE_ASSETS_DIR ${DAWN_ASSETS_DIR}/games/rose)
tool_prefab(${ROSE_ASSETS_DIR}/prefabs/Player.xml) tool_prefab(${ROSE_ASSETS_DIR}/prefabs/Player.xml)
tool_prefab(${ROSE_ASSETS_DIR}/prefabs/Urchin.xml) tool_prefab(${ROSE_ASSETS_DIR}/prefabs/Urchin.xml)
tool_prefab(${ROSE_ASSETS_DIR}/prefabs/Crab.xml) tool_prefab(${ROSE_ASSETS_DIR}/prefabs/Crab.xml)
tool_prefab(${ROSE_ASSETS_DIR}/prefabs/SwordHitbox.xml)

View File

@ -5,11 +5,13 @@
#pragma once #pragma once
#include "scene/SceneItemComponent.hpp" #include "scene/SceneItemComponent.hpp"
#include "scene/components/entity/EntityFaction.hpp"
namespace Dawn { namespace Dawn {
class HurtHazard : public SceneItemComponent { class HurtHazard : public SceneItemComponent {
public: public:
int32_t damage = 1; int32_t damage = 1;
enum Faction faction = FACTION_NONE;
HurtHazard(SceneItem* item); HurtHazard(SceneItem* item);
}; };

View File

@ -14,16 +14,13 @@ PlayerController::PlayerController(SceneItem *item) : SceneItemComponent(item) {
std::vector<SceneItemComponent*> PlayerController::getDependencies() { std::vector<SceneItemComponent*> PlayerController::getDependencies() {
return { return {
(this->characterController = item->getComponent<CharacterController2D>()), (this->entityMove = item->getComponent<EntityMove>()),
(this->entityHealth = item->getComponent<EntityHealth>()), (this->entitySwordAttack = item->getComponent<EntitySwordAttack>())
(this->entityMove = item->getComponent<EntityMove>())
}; };
} }
void PlayerController::onStart() { void PlayerController::onStart() {
assertNotNull(this->characterController);
assertNotNull(this->entityMove); assertNotNull(this->entityMove);
assertNotNull(this->entityHealth);
useEvent([&](float_t delta){ useEvent([&](float_t delta){
// Movement // Movement
@ -31,27 +28,9 @@ void PlayerController::onStart() {
INPUT_BIND_NEGATIVE_X, INPUT_BIND_POSITIVE_X, INPUT_BIND_NEGATIVE_X, INPUT_BIND_POSITIVE_X,
INPUT_BIND_NEGATIVE_Y, INPUT_BIND_POSITIVE_Y INPUT_BIND_NEGATIVE_Y, INPUT_BIND_POSITIVE_Y
); );
if(this->entitySwordAttack != nullptr && getGame()->inputManager.isPressed(INPUT_BIND_ACCEPT)) {
this->entitySwordAttack->attack();
}
}, getScene()->eventSceneUpdate); }, getScene()->eventSceneUpdate);
// Hurt Hazard Processing
useEvent([&](struct CharacterController2DCollisionEventInfo info) {
if(entityHealth->isInvincible()) return;
auto hurtHazard = info.collider->item->getComponent<HurtHazard>();
if(hurtHazard == nullptr) return;
glm::vec2 back = glm::vec2(
info.collider->transform->getLocalPosition().x,
info.collider->transform->getLocalPosition().z
) - glm::vec2(
this->transform->getLocalPosition().x,
this->transform->getLocalPosition().z
);
this->characterController->velocity = back * -hitKnockback;
this->entityHealth->damage({
.amount = hurtHazard->damage
});
}, this->characterController->eventCollision);
} }

View File

@ -4,22 +4,16 @@
// https://opensource.org/licenses/MIT // https://opensource.org/licenses/MIT
#pragma once #pragma once
#include "scene/components/physics/2d/CharacterController2D.hpp"
#include "scene/components/HurtHazard.hpp"
#include "scene/components/entity/EntityHealth.hpp"
#include "scene/components/entity/EntityMove.hpp" #include "scene/components/entity/EntityMove.hpp"
#include "scene/components/entity/EntitySwordAttack.hpp"
namespace Dawn { namespace Dawn {
class PlayerController : public SceneItemComponent { class PlayerController : public SceneItemComponent {
protected: protected:
CharacterController2D *characterController; EntityMove *entityMove = nullptr;
EntityHealth *entityHealth; EntitySwordAttack *entitySwordAttack = nullptr;
EntityMove *entityMove;
public: public:
// @optional
float_t hitKnockback = 20.0f;
PlayerController(SceneItem *item); PlayerController(SceneItem *item);
std::vector<SceneItemComponent*> getDependencies() override; std::vector<SceneItemComponent*> getDependencies() override;
void onStart() override; void onStart() override;

View File

@ -10,4 +10,6 @@ target_sources(${DAWN_TARGET_NAME}
EntityHealth.cpp EntityHealth.cpp
EntityRandomWalk.cpp EntityRandomWalk.cpp
EntityMove.cpp EntityMove.cpp
EntitySwordAttack.cpp
EntityFaction.cpp
) )

View File

@ -13,12 +13,14 @@ EntityChargePlayer::EntityChargePlayer(SceneItem* item) : SceneItemComponent(ite
std::vector<SceneItemComponent*> EntityChargePlayer::getDependencies() { std::vector<SceneItemComponent*> EntityChargePlayer::getDependencies() {
return { return {
(characterController = item->getComponent<CharacterController2D>()) (characterController = item->getComponent<CharacterController2D>()),
(this->entityMove = item->getComponent<EntityMove>())
}; };
} }
void EntityChargePlayer::onStart() { void EntityChargePlayer::onStart() {
assertNotNull(characterController); assertNotNull(characterController);
assertNotNull(this->entityMove);
useEvent([&](float_t delta) { useEvent([&](float_t delta) {
// Get the direction to move in // Get the direction to move in
@ -26,7 +28,7 @@ void EntityChargePlayer::onStart() {
if(player == nullptr) return; if(player == nullptr) return;
// Get the direction to move in // Get the direction to move in
glm::vec2 pos = glm::vec2( glm::vec2 diff = glm::vec2(
player->transform->getLocalPosition().x, player->transform->getLocalPosition().x,
player->transform->getLocalPosition().z player->transform->getLocalPosition().z
) - glm::vec2( ) - glm::vec2(
@ -34,14 +36,10 @@ void EntityChargePlayer::onStart() {
transform->getLocalPosition().z transform->getLocalPosition().z
); );
// Conver to an angle and then convert back into a 2D vector. This mimics a if(glm::distance(glm::vec2(), diff) <= 2.0f) {
// user essentially "moving the analogue stick in this direction" this->entityMove->direction = glm::vec2(0, 0);
} else {
if(mathAbs(pos.x) > 1.0f || mathAbs(pos.y) > 1.0f) { this->entityMove->direction = diff;
float_t angle = atan2(pos.y, pos.x);
glm::vec2 movement(cos(angle), sin(angle));
transform->setLocalRotation(glm::quat(glm::vec3(0, -angle + 1.5708f, 0)));
characterController->velocity += movement * delta * moveSpeed;
} }
}, getScene()->eventSceneUpdate); }, getScene()->eventSceneUpdate);
} }

View File

@ -7,16 +7,15 @@
#include "scene/SceneItemComponent.hpp" #include "scene/SceneItemComponent.hpp"
#include "scene/components/physics/2d/CharacterController2D.hpp" #include "scene/components/physics/2d/CharacterController2D.hpp"
#include "scene/components/PlayerController.hpp" #include "scene/components/PlayerController.hpp"
#include "scene/components/entity/EntityMove.hpp"
namespace Dawn { namespace Dawn {
class EntityChargePlayer : public SceneItemComponent { class EntityChargePlayer : public SceneItemComponent {
protected: protected:
CharacterController2D *characterController = nullptr; CharacterController2D *characterController = nullptr;
EntityMove *entityMove = nullptr;
public: public:
// @optional
float_t moveSpeed = 30.0f;
EntityChargePlayer(SceneItem* item); EntityChargePlayer(SceneItem* item);
std::vector<SceneItemComponent*> getDependencies() override; std::vector<SceneItemComponent*> getDependencies() override;
void onStart() override; void onStart() override;

View File

@ -0,0 +1,12 @@
// Copyright (c) 2023 Dominic Masters
//
// This software is released under the MIT License.
// https://opensource.org/licenses/MIT
#include "EntityFaction.hpp"
using namespace Dawn;
EntityFaction::EntityFaction(SceneItem* item) : SceneItemComponent(item) {
}

View File

@ -0,0 +1,23 @@
// Copyright (c) 2023 Dominic Masters
//
// This software is released under the MIT License.
// https://opensource.org/licenses/MIT
#pragma once
#include "scene/SceneItemComponent.hpp"
namespace Dawn {
enum Faction {
FACTION_NONE,
FACTION_ENEMY,
FACTION_PLAYER
};
class EntityFaction : public SceneItemComponent {
public:
// @optional
enum Faction faction = FACTION_NONE;
EntityFaction(SceneItem* item);
};
}

View File

@ -11,7 +11,15 @@ EntityHealth::EntityHealth(SceneItem* item) : SceneItemComponent(item) {
} }
std::vector<SceneItemComponent*> EntityHealth::getDependencies() {
return {
this->characterController = this->item->getComponent<CharacterController2D>(),
this->entityFaction = this->item->getComponent<EntityFaction>()
};
}
void EntityHealth::onStart() { void EntityHealth::onStart() {
// Update
useEvent([&](float_t delta){ useEvent([&](float_t delta){
if(this->invincibleTime > 0.0f) { if(this->invincibleTime > 0.0f) {
this->invincibleTime -= delta; this->invincibleTime -= delta;
@ -21,6 +29,32 @@ void EntityHealth::onStart() {
this->stunTime -= delta; this->stunTime -= delta;
} }
}, getScene()->eventSceneUpdate); }, getScene()->eventSceneUpdate);
// Hurt Hazard Processing
useEvent([&](struct CharacterController2DCollisionEventInfo info) {
if(this->isInvincible()) return;
auto hurtHazard = info.collider->item->getComponent<HurtHazard>();
if(hurtHazard == nullptr) return;
if(
entityFaction != nullptr &&
hurtHazard->faction == this->entityFaction->faction
) return;
glm::vec2 back = glm::vec2(
info.collider->transform->getLocalPosition().x,
info.collider->transform->getLocalPosition().z
) - glm::vec2(
this->transform->getLocalPosition().x,
this->transform->getLocalPosition().z
);
this->characterController->velocity = back * -hitKnockback;
this->damage({
.amount = hurtHazard->damage
});
}, this->characterController->eventCollision);
} }
bool_t EntityHealth::isInvincible() { bool_t EntityHealth::isInvincible() {

View File

@ -5,6 +5,9 @@
#pragma once #pragma once
#include "scene/SceneItemComponent.hpp" #include "scene/SceneItemComponent.hpp"
#include "scene/components/entity/EntityFaction.hpp"
#include "scene/components/physics/2d/CharacterController2D.hpp"
#include "scene/components/HurtHazard.hpp"
namespace Dawn { namespace Dawn {
struct DamageInformation { struct DamageInformation {
@ -14,6 +17,10 @@ namespace Dawn {
}; };
class EntityHealth : public SceneItemComponent { class EntityHealth : public SceneItemComponent {
protected:
EntityFaction *entityFaction = nullptr;
CharacterController2D *characterController = nullptr;
public: public:
// @optional // @optional
int32_t health = 10; int32_t health = 10;
@ -23,8 +30,11 @@ namespace Dawn {
float_t invincibleTime = 0.0f; float_t invincibleTime = 0.0f;
// @optional // @optional
float_t stunTime = 0.0f; float_t stunTime = 0.0f;
// @optional
float_t hitKnockback = 20.0f;
EntityHealth(SceneItem* item); EntityHealth(SceneItem* item);
std::vector<SceneItemComponent*> getDependencies() override;
void onStart() override; void onStart() override;
/** /**

View File

@ -14,7 +14,8 @@ EntityMove::EntityMove(SceneItem* item) : SceneItemComponent(item) {
std::vector<SceneItemComponent*> EntityMove::getDependencies() { std::vector<SceneItemComponent*> EntityMove::getDependencies() {
return { return {
(characterController = item->getComponent<CharacterController2D>()), (characterController = item->getComponent<CharacterController2D>()),
(this->entityHealth = item->getComponent<EntityHealth>()) (this->entityHealth = item->getComponent<EntityHealth>()),
(this->entitySwordAttack = item->getComponent<EntitySwordAttack>())
}; };
} }
@ -23,8 +24,9 @@ void EntityMove::onStart() {
assertNotNull(this->entityHealth); assertNotNull(this->entityHealth);
useEvent([&](float_t delta) { useEvent([&](float_t delta) {
// Don't move if stunned. // Don't move if stunned or attacking.
if(entityHealth->isStunned()) return; if(entityHealth->isStunned()) return;
if(entitySwordAttack != nullptr && entitySwordAttack->isAttacking()) return;
// Don't move if no move. // Don't move if no move.
if(direction.x == 0 && direction.y == 0) return; if(direction.x == 0 && direction.y == 0) return;

View File

@ -7,12 +7,14 @@
#include "scene/SceneItemComponent.hpp" #include "scene/SceneItemComponent.hpp"
#include "scene/components/physics/2d/CharacterController2D.hpp" #include "scene/components/physics/2d/CharacterController2D.hpp"
#include "scene/components/entity/EntityHealth.hpp" #include "scene/components/entity/EntityHealth.hpp"
#include "scene/components/entity/EntitySwordAttack.hpp"
namespace Dawn { namespace Dawn {
class EntityMove : public SceneItemComponent { class EntityMove : public SceneItemComponent {
protected: protected:
CharacterController2D *characterController = nullptr; CharacterController2D *characterController = nullptr;
EntityHealth *entityHealth = nullptr; EntityHealth *entityHealth = nullptr;
EntitySwordAttack *entitySwordAttack = nullptr;
public: public:
// @optional // @optional

View File

@ -0,0 +1,49 @@
// Copyright (c) 2023 Dominic Masters
//
// This software is released under the MIT License.
// https://opensource.org/licenses/MIT
#include "EntitySwordAttack.hpp"
#include "prefabs/SwordHitbox.hpp"
using namespace Dawn;
EntitySwordAttack::EntitySwordAttack(SceneItem* item) :
SceneItemComponent(item),
attackTime(0.0f)
{
}
void EntitySwordAttack::onStart() {
attackHitbox = SwordHitbox::create(getScene());
attackHitbox->transform.setLocalPosition(glm::vec3(999999999, 0, 0));
attackHitbox->transform.setParent(this->transform);
useEvent([&](float_t delta) {
if(this->attackTime > 0.0f) {
this->attackTime = this->attackTime - delta;
}
}, getScene()->eventSceneUpdate);
useEffect([&]{
if(attackTime <= 0) {
attackHitbox->transform.setLocalPosition(glm::vec3(999999999, 0, 0));
} else {
attackHitbox->transform.setLocalPosition(this->attackOffset);
}
}, attackTime);
}
void EntitySwordAttack::attack() {
auto entityFaction = this->item->getComponent<EntityFaction>();
auto hazard = attackHitbox->getComponent<HurtHazard>();
this->attackTime = swingTime;
if(hazard != nullptr && entityFaction != nullptr) {
hazard->faction = entityFaction->faction;
}
}
bool_t EntitySwordAttack::isAttacking() {
return this->attackTime > 0.0f;
}

View File

@ -0,0 +1,30 @@
// Copyright (c) 2023 Dominic Masters
//
// This software is released under the MIT License.
// https://opensource.org/licenses/MIT
#pragma once
#include "scene/SceneItemComponent.hpp"
#include "scene/components/entity/EntityFaction.hpp"
namespace Dawn {
class EntitySwordAttack : public SceneItemComponent {
protected:
SceneItem *attackHitbox = nullptr;
public:
// @optional
StateProperty<float_t> attackTime;
// @optional
glm::vec3 attackOffset = glm::vec3(0, 0, 1);
// @optional
float_t swingTime = 0.5f;
EntitySwordAttack(SceneItem* item);
void onStart() override;
void attack();
bool_t isAttacking();
};
}

View File

@ -59,6 +59,8 @@ int32_t PrefabComponentParser::onParse(
parser = vec3Parser; parser = vec3Parser;
} else if(type == "int32_t" || type == "int") { } else if(type == "int32_t" || type == "int") {
parser = intParser; parser = intParser;
} else if(type.starts_with("enum")) {
parser = rawParser;
} else if(type.find("*") == (type.size() - 1)) { } else if(type.find("*") == (type.size() - 1)) {
type = type.substr(0, type.size() - 1); type = type.substr(0, type.size() - 1);
parser = rawParser; parser = rawParser;
@ -66,7 +68,7 @@ int32_t PrefabComponentParser::onParse(
name = name.substr(1, name.size()); name = name.substr(1, name.size());
parser = rawParser; parser = rawParser;
} else { } else {
*error = "Unkown parser for type: " + type + "::" + name; *error = "Unknown parser for type: " + type + "::" + name;
return 1; return 1;
} }