. */ declare(strict_types=1); namespace App\Repository; use App\Entity\Base\AbstractStructuralDBElement; use App\Helpers\Trees\StructuralDBElementIterator; use App\Helpers\Trees\TreeViewNode; use RecursiveIteratorIterator; class StructuralDBElementRepository extends NamedDBElementRepository { /** * Finds all nodes without a parent node. They are our root nodes. * * @return AbstractStructuralDBElement[] */ public function findRootNodes(): array { return $this->findBy(['parent' => null], ['name' => 'ASC']); } /** * Gets a tree of TreeViewNode elements. The root elements has $parent as parent. * The treeview is generic, that means the href are null and ID values are set. * * @param AbstractStructuralDBElement|null $parent the parent the root elements should have * * @return TreeViewNode[] */ public function getGenericNodeTree(?AbstractStructuralDBElement $parent = null): array { $result = []; $entities = $this->findBy(['parent' => $parent], ['name' => 'ASC']); foreach ($entities as $entity) { /** @var AbstractStructuralDBElement $entity */ //Make a recursive call to find all children nodes $children = $this->getGenericNodeTree($entity); $node = new TreeViewNode($entity->getName(), null, $children); //Set the ID of this entity to later be able to reconstruct the URL $node->setId($entity->getID()); $result[] = $node; } return $result; } /** * Gets a flattened hierarchical tree. Useful for generating option lists. * * @param AbstractStructuralDBElement|null $parent This entity will be used as root element. Set to null, to use global root * * @return AbstractStructuralDBElement[] a flattened list containing the tree elements */ public function toNodesList(?AbstractStructuralDBElement $parent = null): array { $result = []; $entities = $this->findBy(['parent' => $parent], ['name' => 'ASC']); $elementIterator = new StructuralDBElementIterator($entities); $recursiveIterator = new RecursiveIteratorIterator($elementIterator, RecursiveIteratorIterator::SELF_FIRST); //$result = iterator_to_array($recursiveIterator); //We can not use iterator_to_array here or we get only the parent elements foreach ($recursiveIterator as $item) { $result[] = $item; } return $result; } /** * Creates a structure of AbsstractStructuralDBElements from a path separated by $separator, which splits the various levels. * This function will try to use existing elements, if they are already in the database. If not, they will be created. * An array of the created elements will be returned, with the last element being the deepest element. * @param string $path * @param string $separator * @return AbstractStructuralDBElement[] */ public function getNewEntityFromPath(string $path, string $separator = '->'): array { $parent = null; $result = []; foreach (explode($separator, $path) as $name) { $name = trim($name); if ('' === $name) { continue; } //See if we already have an element with this name and parent $entity = $this->findOneBy(['name' => $name, 'parent' => $parent]); if (null === $entity) { $class = $this->getClassName(); /** @var AbstractStructuralDBElement $entity */ $entity = new $class; $entity->setName($name); $entity->setParent($parent); } $result[] = $entity; $parent = $entity; } return $result; } }