Add door interaction system with state management and event publishing
This commit is contained in:
25
GameCore/Interaction/DoorComponent.cs
Normal file
25
GameCore/Interaction/DoorComponent.cs
Normal file
@@ -0,0 +1,25 @@
|
||||
using GameCore.ECS.Interfaces;
|
||||
using GameCore.Interaction.Interfaces;
|
||||
|
||||
namespace GameCore.Interaction;
|
||||
|
||||
public class DoorComponent : IComponent
|
||||
{
|
||||
public enum DoorState
|
||||
{
|
||||
Locked,
|
||||
Closed,
|
||||
Opening,
|
||||
Open,
|
||||
Closing
|
||||
}
|
||||
|
||||
public DoorState CurrentState { get; set; } = DoorState.Locked;
|
||||
public List<IInteractionRequirement> Requirements { get; set; } = [];
|
||||
|
||||
public bool IsOneTimeUnlock { get; set; } = false;
|
||||
|
||||
public float OpenSpeed { get; set; } = 2.0f;
|
||||
public float OpenProgress { get; set; } = 0.0f;
|
||||
public float Timer { get; set; } = 0.0f;
|
||||
}
|
||||
59
GameCore/Interaction/DoorSystem.cs
Normal file
59
GameCore/Interaction/DoorSystem.cs
Normal file
@@ -0,0 +1,59 @@
|
||||
using GameCore.ECS;
|
||||
using GameCore.ECS.Interfaces;
|
||||
using GameCore.Events;
|
||||
|
||||
namespace GameCore.Interaction;
|
||||
|
||||
public class DoorSystem : ISystem
|
||||
{
|
||||
public void Update(World world, float deltaTime)
|
||||
{
|
||||
var doors = world.GetEntitiesWith<DoorComponent>();
|
||||
foreach (var entity in doors)
|
||||
{
|
||||
var door = world.GetComponent<DoorComponent>(entity);
|
||||
if (door == null) continue;
|
||||
|
||||
switch (door.CurrentState)
|
||||
{
|
||||
case DoorComponent.DoorState.Opening:
|
||||
ApplyMovement(entity, door, world, deltaTime, DoorComponent.DoorState.Open,
|
||||
System.Math.Max(door.OpenSpeed, 0f));
|
||||
break;
|
||||
|
||||
case DoorComponent.DoorState.Closing:
|
||||
ApplyMovement(entity, door, world, -deltaTime, DoorComponent.DoorState.Closed, 0f);
|
||||
break;
|
||||
|
||||
case DoorComponent.DoorState.Open:
|
||||
door.Timer = System.Math.Max(door.OpenSpeed, 0f);
|
||||
door.OpenProgress = 1f;
|
||||
break;
|
||||
|
||||
case DoorComponent.DoorState.Closed:
|
||||
case DoorComponent.DoorState.Locked:
|
||||
door.Timer = 0f;
|
||||
door.OpenProgress = 0f;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private void ApplyMovement(Entity entity, DoorComponent door, World world, float delta,
|
||||
DoorComponent.DoorState targetState, float targetBoundary)
|
||||
{
|
||||
var effectiveSpeed = System.Math.Max(door.OpenSpeed, 0f);
|
||||
door.Timer = System.Math.Clamp(door.Timer + delta, 0f, effectiveSpeed);
|
||||
|
||||
if (effectiveSpeed > 0f)
|
||||
door.OpenProgress = System.Math.Clamp(door.Timer / effectiveSpeed, 0f, 1f);
|
||||
else
|
||||
door.OpenProgress = door.Timer > 0f ? 1f : 0f;
|
||||
|
||||
if ((!(delta > 0f) || !(door.Timer >= targetBoundary)) &&
|
||||
(!(delta < 0f) || !(door.Timer <= targetBoundary))) return;
|
||||
|
||||
door.CurrentState = targetState;
|
||||
world.PublishEvent(new DoorStateChangedEvent(entity, door.CurrentState));
|
||||
}
|
||||
}
|
||||
80
GameCore/Interaction/InteractionSystem.cs
Normal file
80
GameCore/Interaction/InteractionSystem.cs
Normal file
@@ -0,0 +1,80 @@
|
||||
using GameCore.ECS;
|
||||
using GameCore.ECS.Interfaces;
|
||||
using GameCore.Events;
|
||||
using GameCore.Input;
|
||||
using GameCore.Player;
|
||||
|
||||
namespace GameCore.Interaction;
|
||||
|
||||
public class InteractionSystem(float interactionRange) : ISystem
|
||||
{
|
||||
public void Update(World world, float deltaTime)
|
||||
{
|
||||
var playerEntities = world.GetEntitiesWith<PlayerComponent>();
|
||||
var players = playerEntities.ToList();
|
||||
if (!players.Any()) return;
|
||||
|
||||
var player = players.First();
|
||||
var input = world.GetComponent<InputStateComponent>(player);
|
||||
if (input == null) return;
|
||||
|
||||
world.RemoveComponent<IsLookingAtInteractableComponent>(player);
|
||||
|
||||
var from = input.MuzzlePosition;
|
||||
var to = from + input.MuzzleDirection * interactionRange;
|
||||
var hit = world.WorldQuery.Raycast(from, to, player);
|
||||
|
||||
if (hit.DidHit && hit.HitEntity.HasValue)
|
||||
{
|
||||
var door = world.GetComponent<DoorComponent>(hit.HitEntity.Value);
|
||||
if (door != null)
|
||||
{
|
||||
world.AddComponent(player, new IsLookingAtInteractableComponent(hit.HitEntity.Value));
|
||||
|
||||
if (input.IsInteracting) TryInteractWithDoor(world, player, hit.HitEntity.Value, door);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private void TryInteractWithDoor(World world, Entity interactor, Entity doorEntity, DoorComponent door)
|
||||
{
|
||||
switch (door.CurrentState)
|
||||
{
|
||||
case DoorComponent.DoorState.Locked:
|
||||
if (CheckRequirements(world, interactor, door))
|
||||
{
|
||||
world.Logger.Info($"Door {doorEntity.Id} requirements met. Unlocking door.");
|
||||
door.CurrentState = DoorComponent.DoorState.Opening;
|
||||
if (door.IsOneTimeUnlock) door.Requirements.Clear();
|
||||
|
||||
world.PublishEvent(new DoorStateChangedEvent(doorEntity, door.CurrentState));
|
||||
}
|
||||
else
|
||||
{
|
||||
world.Logger.Info($"Door {doorEntity.Id} requirements not met. Cannot unlock door.");
|
||||
world.PublishEvent(new DoorLockedEvent(doorEntity, interactor));
|
||||
}
|
||||
|
||||
break;
|
||||
case DoorComponent.DoorState.Closed:
|
||||
door.CurrentState = DoorComponent.DoorState.Opening;
|
||||
world.PublishEvent(new DoorStateChangedEvent(doorEntity, door.CurrentState));
|
||||
break;
|
||||
case DoorComponent.DoorState.Open:
|
||||
door.CurrentState = DoorComponent.DoorState.Closing;
|
||||
world.PublishEvent(new DoorStateChangedEvent(doorEntity, door.CurrentState));
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
private bool CheckRequirements(World world, Entity interactor, DoorComponent door)
|
||||
{
|
||||
foreach (var req in door.Requirements)
|
||||
if (!req.IsMet(interactor, world))
|
||||
return false;
|
||||
|
||||
foreach (var req in door.Requirements) req.ApplySideEffects(interactor, world);
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,9 @@
|
||||
using GameCore.ECS;
|
||||
|
||||
namespace GameCore.Interaction.Interfaces;
|
||||
|
||||
public interface IInteractionRequirement
|
||||
{
|
||||
bool IsMet(Entity interactor, World world);
|
||||
void ApplySideEffects(Entity interactor, World world);
|
||||
}
|
||||
9
GameCore/Interaction/IsLookingAtInteractableComponent.cs
Normal file
9
GameCore/Interaction/IsLookingAtInteractableComponent.cs
Normal file
@@ -0,0 +1,9 @@
|
||||
using GameCore.ECS;
|
||||
using GameCore.ECS.Interfaces;
|
||||
|
||||
namespace GameCore.Interaction;
|
||||
|
||||
public class IsLookingAtInteractableComponent(Entity target) : IComponent
|
||||
{
|
||||
public readonly Entity Target = target;
|
||||
}
|
||||
37
GameCore/Interaction/RequiresAttributeRequirement.cs
Normal file
37
GameCore/Interaction/RequiresAttributeRequirement.cs
Normal file
@@ -0,0 +1,37 @@
|
||||
using GameCore.Attributes;
|
||||
using GameCore.ECS;
|
||||
using GameCore.Interaction.Interfaces;
|
||||
using Attribute = GameCore.Attributes.Attribute;
|
||||
|
||||
namespace GameCore.Interaction;
|
||||
|
||||
public enum ComparisonType
|
||||
{
|
||||
GreaterOrEqual = 0,
|
||||
LessThan = 1,
|
||||
Equal = 2,
|
||||
}
|
||||
|
||||
public class RequiresAttributeRequirement(Attribute attribute, float value, ComparisonType comparison)
|
||||
: IInteractionRequirement
|
||||
{
|
||||
public bool IsMet(Entity interactor, World world)
|
||||
{
|
||||
var attributes = world.GetComponent<AttributeComponent>(interactor);
|
||||
if (attributes == null) return false;
|
||||
|
||||
var actualValue = attributes.GetValue(attribute);
|
||||
|
||||
return comparison switch
|
||||
{
|
||||
ComparisonType.GreaterOrEqual => actualValue >= value,
|
||||
ComparisonType.LessThan => actualValue < value,
|
||||
ComparisonType.Equal => System.Math.Abs(actualValue - value) < 0.0001f,
|
||||
_ => false
|
||||
};
|
||||
}
|
||||
|
||||
public void ApplySideEffects(Entity interactor, World world)
|
||||
{
|
||||
}
|
||||
}
|
||||
26
GameCore/Interaction/RequiresItemRequirement.cs
Normal file
26
GameCore/Interaction/RequiresItemRequirement.cs
Normal file
@@ -0,0 +1,26 @@
|
||||
using GameCore.ECS;
|
||||
using GameCore.Events;
|
||||
using GameCore.Interaction.Interfaces;
|
||||
using GameCore.Inventory;
|
||||
|
||||
namespace GameCore.Interaction;
|
||||
|
||||
public class RequiresItemRequirement(string itemId, int quantity, bool consumeItem) : IInteractionRequirement
|
||||
{
|
||||
public bool IsMet(Entity interactor, World world)
|
||||
{
|
||||
var inventory = world.GetComponent<InventoryComponent>(interactor);
|
||||
return inventory != null && inventory.HasItem(itemId, quantity);
|
||||
}
|
||||
|
||||
public void ApplySideEffects(Entity interactor, World world)
|
||||
{
|
||||
if (!consumeItem) return;
|
||||
|
||||
var inventory = world.GetComponent<InventoryComponent>(interactor);
|
||||
if (inventory == null || !inventory.RemoveItem(itemId, quantity)) return;
|
||||
|
||||
var newQuantity = inventory.GetItemCount(itemId);
|
||||
world.PublishEvent(new InventoryItemChangedEvent(interactor, itemId, newQuantity));
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user