Files
doom-dm/EngineTests/src/Scene/HierarchyTests.cs
2025-01-02 05:15:16 +03:00

548 lines
14 KiB
C#

using Engine.Scene;
namespace EngineTests.Scene;
public class HierarchyTests
{
private Hierarchy<object> _hierarchy;
[SetUp]
public void Setup()
{
_hierarchy = new Hierarchy<object>();
}
[Test]
public void AddToRoot()
{
var obj = new object();
_hierarchy.Add(obj);
_hierarchy.ProcessChanges();
Assert.Multiple(() =>
{
Assert.That(_hierarchy.GetChildren().Count(), Is.EqualTo(1));
Assert.That(_hierarchy.GetParent(obj), Is.Null);
});
}
[Test]
public void AddToChild()
{
var parent = new object();
var child = new object();
_hierarchy.Add(parent);
_hierarchy.Add(child);
_hierarchy.AddChild(parent, child);
_hierarchy.ProcessChanges();
Assert.Multiple(() =>
{
Assert.That(_hierarchy.GetChildren(parent).Count(), Is.EqualTo(1));
Assert.That(_hierarchy.GetParent(parent), Is.Null);
Assert.That(_hierarchy.GetParent(child), Is.EqualTo(parent));
});
}
[Test]
public void RemoveChild()
{
var parent = new object();
var child = new object();
_hierarchy.Add(parent);
_hierarchy.Add(child);
_hierarchy.AddChild(parent, child);
_hierarchy.ProcessChanges();
_hierarchy.Remove(child);
_hierarchy.ProcessChanges();
Assert.Multiple(() =>
{
Assert.That(_hierarchy.GetChildren(parent).Count(), Is.EqualTo(0));
Assert.That(_hierarchy.Contains(child), Is.False);
});
}
[Test]
public void RemoveParentAndAllChildren()
{
var parent = new object();
var child1 = new object();
var child2 = new object();
_hierarchy.Add(parent);
_hierarchy.Add(child1);
_hierarchy.Add(child2);
_hierarchy.AddChild(parent, child1);
_hierarchy.AddChild(parent, child2);
_hierarchy.ProcessChanges();
_hierarchy.Remove(parent);
_hierarchy.ProcessChanges();
Assert.Multiple(() =>
{
Assert.That(_hierarchy.GetChildren().Count(), Is.EqualTo(0));
Assert.That(_hierarchy.Contains(child1), Is.False);
Assert.That(_hierarchy.Contains(child2), Is.False);
Assert.That(_hierarchy.Contains(parent), Is.False);
});
}
[Test]
public void DetectCycleInHierarchy()
{
var parent = new object();
var child = new object();
_hierarchy.Add(parent);
_hierarchy.AddChild(parent, child);
_hierarchy.AddChild(child, parent);
Assert.Throws<InvalidOperationException>(() => _hierarchy.ProcessChanges());
}
[Test]
public void IsInHierarchy_PositiveCase()
{
var ancestor = new object();
var child = new object();
_hierarchy.Add(ancestor);
_hierarchy.Add(child);
_hierarchy.AddChild(ancestor, child);
_hierarchy.ProcessChanges();
Assert.That(_hierarchy.IsInHierarchy(ancestor, child), Is.True);
}
[Test]
public void IsInHierarchy_NegativeCase()
{
var obj1 = new object();
var obj2 = new object();
_hierarchy.Add(obj1);
_hierarchy.Add(obj2);
_hierarchy.ProcessChanges();
Assert.That(_hierarchy.IsInHierarchy(obj1, obj2), Is.False);
}
[Test]
public void MultipleChildrenHandling()
{
var parent = new object();
var child1 = new object();
var child2 = new object();
_hierarchy.Add(parent);
_hierarchy.Add(child1);
_hierarchy.Add(child2);
_hierarchy.AddChild(parent, child1);
_hierarchy.AddChild(parent, child2);
_hierarchy.ProcessChanges();
Assert.Multiple(() =>
{
Assert.That(_hierarchy.GetChildren(parent).Count(), Is.EqualTo(2));
Assert.That(_hierarchy.GetParent(child1), Is.EqualTo(parent));
Assert.That(_hierarchy.GetParent(child2), Is.EqualTo(parent));
});
}
[Test]
public void GetChildren_EmptyCase()
{
var obj = new object();
_hierarchy.Add(obj);
_hierarchy.ProcessChanges();
Assert.That(_hierarchy.GetChildren(obj).Count(), Is.EqualTo(0));
}
[Test]
public void AddDuplicateObjectThrows()
{
var obj = new object();
_hierarchy.Add(obj);
_hierarchy.Add(obj);
Assert.Throws<ArgumentException>(() => _hierarchy.ProcessChanges());
}
[Test]
public void AddChild_ParentNotInHierarchyThrows()
{
var parent = new object();
var child = new object();
_hierarchy.Add(child);
_hierarchy.AddChild(parent, child);
Assert.Throws<InvalidOperationException>(() => _hierarchy.ProcessChanges());
}
[Test]
public void GetParent_ObjectNotInHierarchyThrows()
{
var obj = new object();
Assert.That(_hierarchy.Contains(obj), Is.False);
}
[Test]
public void NestedHierarchyTest()
{
var grandParent = new object();
var parent = new object();
var child = new object();
_hierarchy.Add(grandParent);
_hierarchy.Add(parent);
_hierarchy.Add(child);
_hierarchy.AddChild(grandParent, parent);
_hierarchy.AddChild(parent, child);
_hierarchy.ProcessChanges();
Assert.Multiple(() =>
{
Assert.That(_hierarchy.GetParent(parent), Is.EqualTo(grandParent));
Assert.That(_hierarchy.GetParent(child), Is.EqualTo(parent));
Assert.That(_hierarchy.IsInHierarchy(grandParent, child), Is.True);
});
}
[Test]
public void ReparentObjectTest()
{
var parent1 = new object();
var parent2 = new object();
var child = new object();
_hierarchy.Add(parent1);
_hierarchy.Add(parent2);
_hierarchy.Add(child);
_hierarchy.AddChild(parent1, child);
_hierarchy.ProcessChanges();
_hierarchy.AddChild(parent2, child);
_hierarchy.ProcessChanges();
Assert.Multiple(() =>
{
Assert.That(_hierarchy.GetChildren(parent1).Count(), Is.EqualTo(0));
Assert.That(_hierarchy.GetChildren(parent2).Count(), Is.EqualTo(1));
Assert.That(_hierarchy.GetParent(child), Is.EqualTo(parent2));
});
}
[Test]
public void RemoveRootObjectsTest()
{
var obj1 = new object();
var obj2 = new object();
_hierarchy.Add(obj1);
_hierarchy.Add(obj2);
_hierarchy.ProcessChanges();
_hierarchy.Remove(obj1);
_hierarchy.ProcessChanges();
Assert.Multiple(() =>
{
Assert.That(_hierarchy.GetChildren().Count(), Is.EqualTo(1));
Assert.That(_hierarchy.Contains(obj1), Is.False);
});
}
[Test]
public void IsInHierarchy_SelfTest()
{
var obj = new object();
_hierarchy.Add(obj);
_hierarchy.ProcessChanges();
Assert.That(_hierarchy.IsInHierarchy(obj, obj), Is.True);
}
[Test]
public void ComplexHierarchyRemovalTest()
{
var grandParent = new object();
var parent1 = new object();
var parent2 = new object();
var child1 = new object();
var child2 = new object();
_hierarchy.Add(grandParent);
_hierarchy.Add(parent1);
_hierarchy.Add(parent2);
_hierarchy.Add(child1);
_hierarchy.Add(child2);
_hierarchy.AddChild(grandParent, parent1);
_hierarchy.AddChild(grandParent, parent2);
_hierarchy.AddChild(parent1, child1);
_hierarchy.AddChild(parent2, child2);
_hierarchy.ProcessChanges();
_hierarchy.Remove(parent1);
_hierarchy.ProcessChanges();
Assert.Multiple(() =>
{
Assert.That(_hierarchy.GetChildren(grandParent).Count(), Is.EqualTo(1));
Assert.That(_hierarchy.Contains(child1), Is.False);
Assert.That(_hierarchy.Contains(parent1), Is.False);
});
}
[Test]
public void AddChild_SelfAsParentThrowsException()
{
var obj = new object();
_hierarchy.Add(obj);
Assert.Throws<InvalidOperationException>(() => _hierarchy.AddChild(obj, obj));
}
[Test]
public void MultiLevelReparentingTest()
{
var grandParent = new object();
var parent = new object();
var child1 = new object();
var child2 = new object();
_hierarchy.Add(grandParent);
_hierarchy.Add(parent);
_hierarchy.Add(child1);
_hierarchy.Add(child2);
_hierarchy.AddChild(grandParent, parent);
_hierarchy.AddChild(parent, child1);
_hierarchy.AddChild(parent, child2);
_hierarchy.ProcessChanges();
_hierarchy.AddChild(child2, child1);
_hierarchy.ProcessChanges();
Assert.Multiple(() =>
{
Assert.That(_hierarchy.GetChildren(parent).Count(), Is.EqualTo(1));
Assert.That(_hierarchy.GetParent(child1), Is.EqualTo(child2));
Assert.That(_hierarchy.IsInHierarchy(grandParent, child1), Is.True);
});
}
[Test]
public void DeepHierarchyTest()
{
const int hierarchyDepth = 100;
var objects = new object[hierarchyDepth];
for (var i = 0; i < hierarchyDepth; i++)
{
objects[i] = new object();
_hierarchy.Add(objects[i]);
}
for (var i = 1; i < hierarchyDepth; i++)
{
_hierarchy.AddChild(objects[i - 1], objects[i]);
}
_hierarchy.ProcessChanges();
Assert.Multiple(() =>
{
for (var i = 1; i < hierarchyDepth; i++)
{
Assert.That(_hierarchy.GetParent(objects[i]), Is.EqualTo(objects[i - 1]));
Assert.That(_hierarchy.IsInHierarchy(objects[0], objects[i]), Is.True);
}
for (var i = 0; i < hierarchyDepth - 1; i++)
{
var children = _hierarchy.GetChildren(objects[i]).ToList();
Assert.That(children, Has.Count.EqualTo(1));
Assert.That(children[0], Is.EqualTo(objects[i + 1]));
}
});
}
[Test]
public void RemoveFromMiddleOfHierarchyTest()
{
var grandParent = new object();
var parent = new object();
var child1 = new object();
var child2 = new object();
_hierarchy.Add(grandParent);
_hierarchy.Add(parent);
_hierarchy.Add(child1);
_hierarchy.Add(child2);
_hierarchy.AddChild(grandParent, parent);
_hierarchy.AddChild(parent, child1);
_hierarchy.AddChild(parent, child2);
_hierarchy.ProcessChanges();
_hierarchy.Remove(parent);
_hierarchy.ProcessChanges();
Assert.Multiple(() =>
{
Assert.That(_hierarchy.GetChildren(grandParent).Count(), Is.EqualTo(0));
Assert.That(_hierarchy.Contains(child1), Is.False);
Assert.That(_hierarchy.Contains(child2), Is.False);
Assert.That(_hierarchy.Contains(parent), Is.False);
});
}
[Test]
public void ConcurrentModificationTest()
{
var parent = new object();
var children = Enumerable.Range(0, 10)
.Select(_ => new object())
.ToList();
_hierarchy.Add(parent);
children.ForEach(parChild => _hierarchy.Add(parChild));
children.AsParallel().ForAll(parChild => { _hierarchy.AddChild(parent, parChild); });
_hierarchy.ProcessChanges();
Assert.That(_hierarchy.GetChildren(parent).Count(), Is.EqualTo(children.Count));
}
[Test]
public void IsInHierarchy_NullInputs()
{
Assert.Multiple(() =>
{
Assert.That(_hierarchy.IsInHierarchy(new object(), null), Is.False);
Assert.That(_hierarchy.IsInHierarchy(null, new object()), Is.False);
Assert.That(_hierarchy.IsInHierarchy(null, null), Is.False);
});
}
[Test]
public void IsInHierarchy_SameObject()
{
var obj = new object();
_hierarchy.Add(obj);
_hierarchy.ProcessChanges();
Assert.That(_hierarchy.IsInHierarchy(obj, obj), Is.True);
}
[Test]
public void IsInHierarchy_DirectParentChild()
{
var parent = new object();
var child = new object();
_hierarchy.Add(parent);
_hierarchy.Add(child);
_hierarchy.AddChild(parent, child);
_hierarchy.ProcessChanges();
Assert.Multiple(() =>
{
Assert.That(_hierarchy.IsInHierarchy(parent, child), Is.True);
Assert.That(_hierarchy.IsInHierarchy(child, parent), Is.False);
});
}
[Test]
public void IsInHierarchy_DeepHierarchy()
{
var grandParent = new object();
var parent = new object();
var child = new object();
var grandChild = new object();
_hierarchy.Add(grandParent);
_hierarchy.Add(parent);
_hierarchy.Add(child);
_hierarchy.Add(grandChild);
_hierarchy.AddChild(grandParent, parent);
_hierarchy.AddChild(parent, child);
_hierarchy.AddChild(child, grandChild);
_hierarchy.ProcessChanges();
Assert.Multiple(() =>
{
Assert.That(_hierarchy.IsInHierarchy(grandParent, grandChild), Is.True);
Assert.That(_hierarchy.IsInHierarchy(parent, grandChild), Is.True);
Assert.That(_hierarchy.IsInHierarchy(child, grandChild), Is.True);
Assert.That(_hierarchy.IsInHierarchy(grandChild, grandParent), Is.False);
Assert.That(_hierarchy.IsInHierarchy(grandChild, parent), Is.False);
Assert.That(_hierarchy.IsInHierarchy(grandChild, child), Is.False);
});
}
[Test]
public void IsInHierarchy_SeparateBranches()
{
var root = new object();
var branch1Parent = new object();
var branch1Child = new object();
var branch2Parent = new object();
var branch2Child = new object();
_hierarchy.Add(root);
_hierarchy.Add(branch1Parent);
_hierarchy.Add(branch1Child);
_hierarchy.Add(branch2Parent);
_hierarchy.Add(branch2Child);
_hierarchy.AddChild(root, branch1Parent);
_hierarchy.AddChild(root, branch2Parent);
_hierarchy.AddChild(branch1Parent, branch1Child);
_hierarchy.AddChild(branch2Parent, branch2Child);
_hierarchy.ProcessChanges();
Assert.Multiple(() =>
{
Assert.That(_hierarchy.IsInHierarchy(root, branch1Parent), Is.True);
Assert.That(_hierarchy.IsInHierarchy(root, branch1Child), Is.True);
Assert.That(_hierarchy.IsInHierarchy(branch1Parent, branch1Child), Is.True);
Assert.That(_hierarchy.IsInHierarchy(branch1Parent, branch2Child), Is.False);
Assert.That(_hierarchy.IsInHierarchy(branch2Parent, branch1Child), Is.False);
});
}
[Test]
public void IsInHierarchy_ObjectsNotInHierarchy()
{
var obj1 = new object();
var obj2 = new object();
_hierarchy.Add(obj1);
_hierarchy.Add(obj2);
_hierarchy.ProcessChanges();
Assert.That(_hierarchy.IsInHierarchy(obj1, obj2), Is.False);
}
}