2017-08-18 22:57:27 +02:00
|
|
|
<?php declare(strict_types=1);
|
2017-01-29 23:20:53 +01:00
|
|
|
|
|
|
|
namespace PhpParser;
|
|
|
|
|
|
|
|
use PhpParser\NodeVisitor\FindingVisitor;
|
|
|
|
use PhpParser\NodeVisitor\FirstFindingVisitor;
|
|
|
|
|
2022-08-28 22:57:06 +02:00
|
|
|
class NodeFinder {
|
2017-01-29 23:20:53 +01:00
|
|
|
/**
|
|
|
|
* Find all nodes satisfying a filter callback.
|
|
|
|
*
|
2023-09-17 15:59:04 +02:00
|
|
|
* @param Node|Node[] $nodes Single node or array of nodes to search in
|
|
|
|
* @param callable $filter Filter callback: function(Node $node) : bool
|
2017-01-29 23:20:53 +01:00
|
|
|
*
|
|
|
|
* @return Node[] Found nodes satisfying the filter callback
|
|
|
|
*/
|
2022-08-28 22:57:06 +02:00
|
|
|
public function find($nodes, callable $filter): array {
|
2023-10-07 17:08:14 +07:00
|
|
|
if ($nodes === []) {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
2017-01-29 23:20:53 +01:00
|
|
|
if (!is_array($nodes)) {
|
|
|
|
$nodes = [$nodes];
|
|
|
|
}
|
|
|
|
|
|
|
|
$visitor = new FindingVisitor($filter);
|
|
|
|
|
2023-07-09 15:34:07 +02:00
|
|
|
$traverser = new NodeTraverser($visitor);
|
2017-01-29 23:20:53 +01:00
|
|
|
$traverser->traverse($nodes);
|
|
|
|
|
|
|
|
return $visitor->getFoundNodes();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Find all nodes that are instances of a certain class.
|
2022-08-08 21:53:49 +02:00
|
|
|
|
|
|
|
* @template TNode as Node
|
2017-01-29 23:20:53 +01:00
|
|
|
*
|
2023-09-17 15:59:04 +02:00
|
|
|
* @param Node|Node[] $nodes Single node or array of nodes to search in
|
2022-08-08 21:53:49 +02:00
|
|
|
* @param class-string<TNode> $class Class name
|
2017-01-29 23:20:53 +01:00
|
|
|
*
|
2023-09-17 15:59:04 +02:00
|
|
|
* @return TNode[] Found nodes (all instances of $class)
|
2017-01-29 23:20:53 +01:00
|
|
|
*/
|
2022-08-28 22:57:06 +02:00
|
|
|
public function findInstanceOf($nodes, string $class): array {
|
2017-01-29 23:20:53 +01:00
|
|
|
return $this->find($nodes, function ($node) use ($class) {
|
|
|
|
return $node instanceof $class;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Find first node satisfying a filter callback.
|
|
|
|
*
|
2023-09-17 15:59:04 +02:00
|
|
|
* @param Node|Node[] $nodes Single node or array of nodes to search in
|
|
|
|
* @param callable $filter Filter callback: function(Node $node) : bool
|
2017-01-29 23:20:53 +01:00
|
|
|
*
|
|
|
|
* @return null|Node Found node (or null if none found)
|
|
|
|
*/
|
2022-06-04 12:48:12 +02:00
|
|
|
public function findFirst($nodes, callable $filter): ?Node {
|
2023-10-07 17:08:14 +07:00
|
|
|
if ($nodes === []) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2017-01-29 23:20:53 +01:00
|
|
|
if (!is_array($nodes)) {
|
|
|
|
$nodes = [$nodes];
|
|
|
|
}
|
|
|
|
|
|
|
|
$visitor = new FirstFindingVisitor($filter);
|
|
|
|
|
2023-07-09 15:34:07 +02:00
|
|
|
$traverser = new NodeTraverser($visitor);
|
2017-01-29 23:20:53 +01:00
|
|
|
$traverser->traverse($nodes);
|
|
|
|
|
|
|
|
return $visitor->getFoundNode();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Find first node that is an instance of a certain class.
|
|
|
|
*
|
2022-08-08 21:53:49 +02:00
|
|
|
* @template TNode as Node
|
|
|
|
*
|
2023-09-17 15:59:04 +02:00
|
|
|
* @param Node|Node[] $nodes Single node or array of nodes to search in
|
2022-08-08 21:53:49 +02:00
|
|
|
* @param class-string<TNode> $class Class name
|
2017-01-29 23:20:53 +01:00
|
|
|
*
|
2023-09-17 15:59:04 +02:00
|
|
|
* @return null|TNode Found node, which is an instance of $class (or null if none found)
|
2017-01-29 23:20:53 +01:00
|
|
|
*/
|
2022-06-04 12:48:12 +02:00
|
|
|
public function findFirstInstanceOf($nodes, string $class): ?Node {
|
2017-01-29 23:20:53 +01:00
|
|
|
return $this->findFirst($nodes, function ($node) use ($class) {
|
|
|
|
return $node instanceof $class;
|
|
|
|
});
|
|
|
|
}
|
2018-01-10 15:04:06 -02:00
|
|
|
}
|