using Ghost.Engine.Components; using Ghost.Entities; namespace Ghost.Editor.Infrastructures.SceneGraph; internal class SceneGraphHelpers { /// /// Creates a new entity with default components. /// /// The world context where the entity will be created. /// The entity to be wrapped in the . public static EntityNode CreateEntityNode(World world, Entity entity, string name) { world.EntityManager.AddComponent(entity, LocalToWorld.Identity); world.EntityManager.AddComponent(entity, Hierarchy.Root); return new EntityNode(entity, name); } /// /// Creates a new and entity with default components. /// /// The world context where the entity will be created. public static EntityNode CreateEntityNode(World world, string name) { var entity = world.EntityManager.CreateEntity(); return CreateEntityNode(world, entity, name); } /// /// Attaches childEntity to parentEntity in the scene graph. /// /// The world context where the entities exist. /// The parent entity to which the child will be attached. /// The child entity to be attached. public static void AttachChild(SceneNode scene, EntityNode parentNode, EntityNode childNode) { // 1) If the child already has a parent, detach it first var childHierarchy = scene.World.EntityManager.GetComponent(childNode.Entity); if (childHierarchy.ValueRO.parent != Entity.Invalid) { DetachFromParent(scene, childNode); } // 2) Link child to new parent childHierarchy.ValueRW.parent = parentNode.Entity; // 3) Insert child at the head of parent's child list var parentHierarchy = scene.World.EntityManager.GetComponent(parentNode.Entity); childHierarchy.ValueRW.nextSibling = parentHierarchy.ValueRO.firstChild; parentHierarchy.ValueRW.firstChild = childNode.Entity; // 4) Write back scene.World.EntityManager.SetComponent(parentNode.Entity, in parentHierarchy.ValueRO); scene.World.EntityManager.SetComponent(childNode.Entity, in childHierarchy.ValueRO); // 5) Update children list in parent node parentNode.Children.Add(childNode); } /// /// Detaches the specified entity from its parent in the scene graph. /// /// The world context where the entities exist. /// The entity to detach from its parent. public static void DetachFromParent(SceneNode scene, EntityNode node) { var hierarchy = scene.World.EntityManager.GetComponent(node.Entity); var parent = hierarchy.ValueRO.parent; if (parent == Entity.Invalid) { return; // already root } var parentHierarchy = scene.World.EntityManager.GetComponent(parent); // If entity is the first child, simply move head if (parentHierarchy.ValueRO.firstChild == node.Entity) { parentHierarchy.ValueRW.firstChild = hierarchy.ValueRO.nextSibling; } else { // Otherwise, find the previous sibling in the linked list var prevSibling = parentHierarchy.ValueRO.firstChild; while (prevSibling != Entity.Invalid) { var prevHierarchy = scene.World.EntityManager.GetComponent(prevSibling); if (prevHierarchy.ValueRW.nextSibling == node.Entity) { prevHierarchy.ValueRW.nextSibling = hierarchy.ValueRO.nextSibling; scene.World.EntityManager.SetComponent(prevSibling, in prevHierarchy.ValueRO); break; } prevSibling = prevHierarchy.ValueRO.nextSibling; } } // Clear child's references hierarchy.ValueRW.parent = Entity.Invalid; hierarchy.ValueRW.nextSibling = Entity.Invalid; // Write back scene.World.EntityManager.SetComponent(parent, in parentHierarchy.ValueRO); scene.World.EntityManager.SetComponent(node.Entity, in hierarchy.ValueRO); // Remove from parent's children list scene.EntityNodeLookup[parent].Children.Remove(node); } }