relations can now contain data

pull/2/head
cosmonaut 2022-04-19 12:35:21 -07:00
parent 833c60003b
commit 5243259acb
7 changed files with 54 additions and 49 deletions

View File

@ -37,7 +37,7 @@ namespace MoonTools.ECS
{ {
// TODO: is it possible to optimize this? // TODO: is it possible to optimize this?
Register<TComponent>(); Register<TComponent>();
return storages[typeof(TComponent)] as ComponentStorage<TComponent>; return (ComponentStorage<TComponent>) storages[typeof(TComponent)];
} }
public bool Some<TComponent>() where TComponent : struct public bool Some<TComponent>() where TComponent : struct

View File

@ -60,22 +60,22 @@ namespace MoonTools.ECS
return EntityStorage.Exists(entity); return EntityStorage.Exists(entity);
} }
protected IEnumerable<Relation> Relations<TRelationKind>() protected IEnumerable<(Entity, Entity, TRelationKind)> Relations<TRelationKind>() where TRelationKind : struct
{ {
return RelationDepot.Relations<TRelationKind>(); return RelationDepot.Relations<TRelationKind>();
} }
protected bool Related<TRelationKind>(in Entity a, in Entity b) protected bool Related<TRelationKind>(in Entity a, in Entity b) where TRelationKind : struct
{ {
return RelationDepot.Related<TRelationKind>(a.ID, b.ID); return RelationDepot.Related<TRelationKind>(a.ID, b.ID);
} }
protected IEnumerable<Entity> RelatedToA<TRelationKind>(in Entity entity) protected IEnumerable<(Entity, TRelationKind)> RelatedToA<TRelationKind>(in Entity entity) where TRelationKind : struct
{ {
return RelationDepot.RelatedToA<TRelationKind>(entity.ID); return RelationDepot.RelatedToA<TRelationKind>(entity.ID);
} }
protected IEnumerable<Entity> RelatedToB<TRelationKind>(in Entity entity) protected IEnumerable<(Entity, TRelationKind)> RelatedToB<TRelationKind>(in Entity entity) where TRelationKind : struct
{ {
return RelationDepot.RelatedToB<TRelationKind>(entity.ID); return RelationDepot.RelatedToB<TRelationKind>(entity.ID);
} }

View File

@ -2,7 +2,7 @@
namespace MoonTools.ECS namespace MoonTools.ECS
{ {
public struct Relation : IEquatable<Relation> internal struct Relation : IEquatable<Relation>
{ {
public Entity A { get; } public Entity A { get; }
public Entity B { get; } public Entity B { get; }
@ -26,12 +26,12 @@ namespace MoonTools.ECS
public bool Equals(Relation other) public bool Equals(Relation other)
{ {
return A.Equals(other.A) && B.Equals(other.B); return A.ID == other.A.ID && B.ID == other.B.ID;
} }
public override int GetHashCode() public override int GetHashCode()
{ {
return HashCode.Combine(A, B); return HashCode.Combine(A.ID, B.ID);
} }
} }
} }

View File

@ -7,26 +7,31 @@ namespace MoonTools.ECS
{ {
private Dictionary<Type, RelationStorage> storages = new Dictionary<Type, RelationStorage>(); private Dictionary<Type, RelationStorage> storages = new Dictionary<Type, RelationStorage>();
private RelationStorage Lookup<TRelationKind>() private void Register<TRelationKind>() where TRelationKind : struct
{ {
return storages[typeof(TRelationKind)]; if (!storages.ContainsKey(typeof(TRelationKind)))
{
storages.Add(typeof(TRelationKind), new RelationStorage<TRelationKind>());
}
} }
public void Register<TRelationKind>() private RelationStorage<TRelationKind> Lookup<TRelationKind>() where TRelationKind : struct
{ {
storages[typeof(TRelationKind)] = new RelationStorage(); Register<TRelationKind>();
return (RelationStorage<TRelationKind>) storages[typeof(TRelationKind)];
} }
public void Add<TRelationKind>(Relation relation) public void Add<TRelationKind>(Relation relation, TRelationKind relationData) where TRelationKind : struct
{ {
Lookup<TRelationKind>().Add(relation); Lookup<TRelationKind>().Add(relation, relationData);
} }
public void Remove<TRelationKind>(Relation relation) public void Remove<TRelationKind>(Relation relation) where TRelationKind : struct
{ {
Lookup<TRelationKind>().Remove(relation); Lookup<TRelationKind>().Remove(relation);
} }
// FIXME: optimize this
public void OnEntityDestroy(int entityID) public void OnEntityDestroy(int entityID)
{ {
foreach (var storage in storages.Values) foreach (var storage in storages.Values)
@ -35,22 +40,22 @@ namespace MoonTools.ECS
} }
} }
public IEnumerable<Relation> Relations<TRelationKind>() public IEnumerable<(Entity, Entity, TRelationKind)> Relations<TRelationKind>() where TRelationKind : struct
{ {
return Lookup<TRelationKind>().All(); return Lookup<TRelationKind>().All();
} }
public bool Related<TRelationKind>(int idA, int idB) public bool Related<TRelationKind>(int idA, int idB) where TRelationKind : struct
{ {
return Lookup<TRelationKind>().Has(new Relation(idA, idB)); return Lookup<TRelationKind>().Has(new Relation(idA, idB));
} }
public IEnumerable<Entity> RelatedToA<TRelationKind>(int entityID) public IEnumerable<(Entity, TRelationKind)> RelatedToA<TRelationKind>(int entityID) where TRelationKind : struct
{ {
return Lookup<TRelationKind>().RelatedToA(entityID); return Lookup<TRelationKind>().RelatedToA(entityID);
} }
public IEnumerable<Entity> RelatedToB<TRelationKind>(int entityID) public IEnumerable<(Entity, TRelationKind)> RelatedToB<TRelationKind>(int entityID) where TRelationKind : struct
{ {
return Lookup<TRelationKind>().RelatedToB(entityID); return Lookup<TRelationKind>().RelatedToB(entityID);
} }

View File

@ -2,24 +2,31 @@
namespace MoonTools.ECS namespace MoonTools.ECS
{ {
internal class RelationStorage internal abstract class RelationStorage
{ {
private HashSet<Relation> relations = new HashSet<Relation>(16); public abstract void OnEntityDestroy(int entityID);
}
// Relation is the two entities, A related to B.
// TRelation is the data attached to the relation.
internal class RelationStorage<TRelation> : RelationStorage where TRelation : struct
{
private Dictionary<Relation, TRelation> relations = new Dictionary<Relation, TRelation>(16);
private Dictionary<int, HashSet<int>> entitiesRelatedToA = new Dictionary<int, HashSet<int>>(16); private Dictionary<int, HashSet<int>> entitiesRelatedToA = new Dictionary<int, HashSet<int>>(16);
private Dictionary<int, HashSet<int>> entitiesRelatedToB = new Dictionary<int, HashSet<int>>(16); private Dictionary<int, HashSet<int>> entitiesRelatedToB = new Dictionary<int, HashSet<int>>(16);
private Stack<HashSet<int>> listPool = new Stack<HashSet<int>>(); private Stack<HashSet<int>> listPool = new Stack<HashSet<int>>();
public IEnumerable<Relation> All() public IEnumerable<(Entity, Entity, TRelation)> All()
{ {
foreach (var relation in relations) foreach (var relationData in relations)
{ {
yield return relation; yield return (relationData.Key.A, relationData.Key.B, relationData.Value);
} }
} }
public void Add(Relation relation) public void Add(Relation relation, TRelation relationData)
{ {
if (relations.Contains(relation)) { return; } if (relations.ContainsKey(relation)) { return; }
var idA = relation.A.ID; var idA = relation.A.ID;
var idB = relation.B.ID; var idB = relation.B.ID;
@ -36,32 +43,35 @@ namespace MoonTools.ECS
} }
entitiesRelatedToB[idB].Add(idA); entitiesRelatedToB[idB].Add(idA);
relations.Add(relation); relations.Add(relation, relationData);
} }
public bool Has(Relation relation) public bool Has(Relation relation)
{ {
return relations.Contains(relation); return relations.ContainsKey(relation);
} }
public IEnumerable<Entity> RelatedToA(int entityID) // FIXME: is there a more descriptive name for these?
public IEnumerable<(Entity, TRelation)> RelatedToA(int entityID)
{ {
if (entitiesRelatedToA.ContainsKey(entityID)) if (entitiesRelatedToA.ContainsKey(entityID))
{ {
foreach (var id in entitiesRelatedToA[entityID]) foreach (var id in entitiesRelatedToA[entityID])
{ {
yield return new Entity(id); var relation = new Relation(entityID, id);
yield return (relation.B, relations[relation]);
} }
} }
} }
public IEnumerable<Entity> RelatedToB(int entityID) public IEnumerable<(Entity, TRelation)> RelatedToB(int entityID)
{ {
if (entitiesRelatedToB.ContainsKey(entityID)) if (entitiesRelatedToB.ContainsKey(entityID))
{ {
foreach (var id in entitiesRelatedToB[entityID]) foreach (var id in entitiesRelatedToB[entityID])
{ {
yield return new Entity(id); var relation = new Relation(id, entityID);
yield return (relation.A, relations[relation]);
} }
} }
} }
@ -81,19 +91,13 @@ namespace MoonTools.ECS
return relations.Remove(relation); return relations.Remove(relation);
} }
// this exists so we don't recurse in OnEntityDestroy public override void OnEntityDestroy(int entityID)
private bool DestroyRemove(Relation relation)
{
return relations.Remove(relation);
}
public void OnEntityDestroy(int entityID)
{ {
if (entitiesRelatedToA.ContainsKey(entityID)) if (entitiesRelatedToA.ContainsKey(entityID))
{ {
foreach (var entityB in entitiesRelatedToA[entityID]) foreach (var entityB in entitiesRelatedToA[entityID])
{ {
DestroyRemove(new Relation(entityID, entityB)); Remove(new Relation(entityID, entityB));
} }
ReturnHashSetToPool(entitiesRelatedToA[entityID]); ReturnHashSetToPool(entitiesRelatedToA[entityID]);
@ -104,7 +108,7 @@ namespace MoonTools.ECS
{ {
foreach (var entityA in entitiesRelatedToB[entityID]) foreach (var entityA in entitiesRelatedToB[entityID])
{ {
DestroyRemove(new Relation(entityA, entityID)); Remove(new Relation(entityA, entityID));
} }
ReturnHashSetToPool(entitiesRelatedToB[entityID]); ReturnHashSetToPool(entitiesRelatedToB[entityID]);
@ -124,6 +128,7 @@ namespace MoonTools.ECS
private void ReturnHashSetToPool(HashSet<int> hashSet) private void ReturnHashSetToPool(HashSet<int> hashSet)
{ {
hashSet.Clear();
listPool.Push(hashSet); listPool.Push(hashSet);
} }
} }

View File

@ -76,12 +76,12 @@ namespace MoonTools.ECS
MessageDepot.Add(message); MessageDepot.Add(message);
} }
protected void Relate<TRelationKind>(in Entity entityA, in Entity entityB) protected void Relate<TRelationKind>(in Entity entityA, in Entity entityB, TRelationKind relationData) where TRelationKind : struct
{ {
RelationDepot.Add<TRelationKind>(new Relation(entityA, entityB)); RelationDepot.Add<TRelationKind>(new Relation(entityA, entityB), relationData);
} }
protected void Unrelate<TRelationKind>(in Entity entityA, in Entity entityB) protected void Unrelate<TRelationKind>(in Entity entityA, in Entity entityB) where TRelationKind : struct
{ {
RelationDepot.Remove<TRelationKind>(new Relation(entityA, entityB)); RelationDepot.Remove<TRelationKind>(new Relation(entityA, entityB));
} }

View File

@ -22,11 +22,6 @@
renderer.RegisterRelationDepot(RelationDepot); renderer.RegisterRelationDepot(RelationDepot);
} }
public void AddRelationKind<TRelationKind>()
{
RelationDepot.Register<TRelationKind>();
}
public Entity CreateEntity() public Entity CreateEntity()
{ {
return EntityStorage.Create(); return EntityStorage.Create();