2020-07-11 11:31:43 +02:00
|
|
|
<?php declare(strict_types=1);
|
|
|
|
|
|
|
|
namespace PhpParser\NodeVisitor;
|
|
|
|
|
|
|
|
use PhpParser\Node;
|
|
|
|
use PhpParser\NodeVisitorAbstract;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Visitor that connects a child node to its parent node
|
|
|
|
* as well as its sibling nodes.
|
|
|
|
*
|
|
|
|
* On the child node, the parent node can be accessed through
|
|
|
|
* <code>$node->getAttribute('parent')</code>, the previous
|
|
|
|
* node can be accessed through <code>$node->getAttribute('previous')</code>,
|
|
|
|
* and the next node can be accessed through <code>$node->getAttribute('next')</code>.
|
|
|
|
*/
|
2022-08-28 22:57:06 +02:00
|
|
|
final class NodeConnectingVisitor extends NodeVisitorAbstract {
|
2020-07-11 11:31:43 +02:00
|
|
|
/**
|
|
|
|
* @var Node[]
|
|
|
|
*/
|
2023-08-16 21:18:30 +02:00
|
|
|
private array $stack = [];
|
2020-07-11 11:31:43 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @var ?Node
|
|
|
|
*/
|
|
|
|
private $previous;
|
|
|
|
|
|
|
|
public function beforeTraverse(array $nodes) {
|
|
|
|
$this->stack = [];
|
|
|
|
$this->previous = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function enterNode(Node $node) {
|
|
|
|
if (!empty($this->stack)) {
|
|
|
|
$node->setAttribute('parent', $this->stack[count($this->stack) - 1]);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($this->previous !== null && $this->previous->getAttribute('parent') === $node->getAttribute('parent')) {
|
|
|
|
$node->setAttribute('previous', $this->previous);
|
|
|
|
$this->previous->setAttribute('next', $node);
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->stack[] = $node;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function leaveNode(Node $node) {
|
|
|
|
$this->previous = $node;
|
|
|
|
|
|
|
|
array_pop($this->stack);
|
|
|
|
}
|
|
|
|
}
|