2024-02-25 15:46:47 +01:00
|
|
|
#pragma once
|
2023-06-20 14:33:09 +10:00
|
|
|
|
|
|
|
#include "StarDamage.hpp"
|
|
|
|
#include "StarDamageTypes.hpp"
|
|
|
|
|
|
|
|
namespace Star {
|
|
|
|
|
|
|
|
STAR_CLASS(World);
|
|
|
|
STAR_CLASS(Entity);
|
|
|
|
STAR_CLASS(DamageManager);
|
|
|
|
|
|
|
|
struct RemoteHitRequest {
|
|
|
|
ConnectionId destinationConnection() const;
|
|
|
|
|
|
|
|
EntityId causingEntityId;
|
|
|
|
EntityId targetEntityId;
|
|
|
|
DamageRequest damageRequest;
|
|
|
|
};
|
|
|
|
|
|
|
|
DataStream& operator<<(DataStream& ds, RemoteHitRequest const& hitRequest);
|
|
|
|
DataStream& operator>>(DataStream& ds, RemoteHitRequest& hitRequest);
|
|
|
|
|
|
|
|
struct RemoteDamageRequest {
|
|
|
|
ConnectionId destinationConnection() const;
|
|
|
|
|
|
|
|
EntityId causingEntityId;
|
|
|
|
EntityId targetEntityId;
|
|
|
|
DamageRequest damageRequest;
|
|
|
|
};
|
|
|
|
|
|
|
|
DataStream& operator<<(DataStream& ds, RemoteDamageRequest const& damageRequest);
|
|
|
|
DataStream& operator>>(DataStream& ds, RemoteDamageRequest& damageRequest);
|
|
|
|
|
|
|
|
struct RemoteDamageNotification {
|
|
|
|
EntityId sourceEntityId;
|
|
|
|
DamageNotification damageNotification;
|
|
|
|
};
|
|
|
|
|
|
|
|
DataStream& operator<<(DataStream& ds, RemoteDamageNotification const& damageNotification);
|
|
|
|
DataStream& operator>>(DataStream& ds, RemoteDamageNotification& damageNotification);
|
|
|
|
|
|
|
|
// Right now, handles entity -> entity damage and ensures that no repeat damage
|
|
|
|
// is applied within the damage cutoff time from the same causing entity.
|
|
|
|
class DamageManager {
|
|
|
|
public:
|
|
|
|
DamageManager(World* world, ConnectionId connectionId);
|
|
|
|
|
|
|
|
// Notify entities that they have caused damage, apply damage to master
|
|
|
|
// entities, produce damage notifications, and run down damage timeouts.
|
2023-07-21 00:58:49 +10:00
|
|
|
void update(float dt);
|
2023-06-20 14:33:09 +10:00
|
|
|
|
|
|
|
// Incoming RemoteHitRequest and RemoteDamageRequest must have the
|
|
|
|
// destinationConnection equal to the DamageManager's connectionId
|
|
|
|
|
|
|
|
void pushRemoteHitRequest(RemoteHitRequest const& remoteHitRequest);
|
|
|
|
void pushRemoteDamageRequest(RemoteDamageRequest const& remoteDamageRequest);
|
|
|
|
void pushRemoteDamageNotification(RemoteDamageNotification remoteDamageNotification);
|
|
|
|
|
|
|
|
List<RemoteHitRequest> pullRemoteHitRequests();
|
|
|
|
List<RemoteDamageRequest> pullRemoteDamageRequests();
|
|
|
|
List<RemoteDamageNotification> pullRemoteDamageNotifications();
|
|
|
|
|
|
|
|
// Pending *local* notifications. Sum of all notifications either generated
|
|
|
|
// locally or recieved.
|
|
|
|
List<DamageNotification> pullPendingNotifications();
|
|
|
|
|
|
|
|
private:
|
|
|
|
struct EntityDamageEvent {
|
|
|
|
Variant<EntityId, String> timeoutGroup;
|
|
|
|
float timeout;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Searches for and queries for hit to any entity within range of the
|
|
|
|
// damage source. Skips over source.sourceEntityId, if set.
|
|
|
|
SmallList<pair<EntityId, HitType>, 4> queryHit(DamageSource const& source, EntityId causingId) const;
|
|
|
|
|
|
|
|
bool isAuthoritative(EntityPtr const& causingEntity, EntityPtr const& targetEntity);
|
|
|
|
|
|
|
|
void addHitRequest(RemoteHitRequest const& remoteHitRequest);
|
|
|
|
void addDamageRequest(RemoteDamageRequest remoteDamageRequest);
|
|
|
|
void addDamageNotification(RemoteDamageNotification remoteDamageNotification);
|
|
|
|
|
|
|
|
World* m_world;
|
|
|
|
ConnectionId m_connectionId;
|
|
|
|
|
|
|
|
// Maps target entity to all of the recent damage events that entity has
|
|
|
|
// received, to prevent rapidly repeating damage.
|
|
|
|
HashMap<EntityId, List<EntityDamageEvent>> m_recentEntityDamages;
|
|
|
|
|
|
|
|
List<RemoteHitRequest> m_pendingRemoteHitRequests;
|
|
|
|
List<RemoteDamageRequest> m_pendingRemoteDamageRequests;
|
|
|
|
List<RemoteDamageNotification> m_pendingRemoteNotifications;
|
|
|
|
List<DamageNotification> m_pendingNotifications;
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|