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;
|
|
|
|
|
2018-01-10 17:24:26 +01:00
|
|
|
class NodeFinder
|
|
|
|
{
|
2017-01-29 23:20:53 +01:00
|
|
|
/**
|
|
|
|
* Find all nodes satisfying a filter callback.
|
|
|
|
*
|
|
|
|
* @param Node|Node[] $nodes Single node or array of nodes to search in
|
|
|
|
* @param callable $filter Filter callback: function(Node $node) : bool
|
|
|
|
*
|
|
|
|
* @return Node[] Found nodes satisfying the filter callback
|
|
|
|
*/
|
2017-04-28 21:40:59 +02:00
|
|
|
public function find($nodes, callable $filter) : array {
|
2017-01-29 23:20:53 +01:00
|
|
|
if (!is_array($nodes)) {
|
|
|
|
$nodes = [$nodes];
|
|
|
|
}
|
|
|
|
|
|
|
|
$visitor = new FindingVisitor($filter);
|
|
|
|
|
|
|
|
$traverser = new NodeTraverser;
|
|
|
|
$traverser->addVisitor($visitor);
|
|
|
|
$traverser->traverse($nodes);
|
|
|
|
|
|
|
|
return $visitor->getFoundNodes();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Find all nodes that are instances of a certain class.
|
|
|
|
*
|
|
|
|
* @param Node|Node[] $nodes Single node or array of nodes to search in
|
|
|
|
* @param string $class Class name
|
|
|
|
*
|
|
|
|
* @return Node[] Found nodes (all instances of $class)
|
|
|
|
*/
|
2017-04-28 21:40:59 +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.
|
|
|
|
*
|
|
|
|
* @param Node|Node[] $nodes Single node or array of nodes to search in
|
|
|
|
* @param callable $filter Filter callback: function(Node $node) : bool
|
|
|
|
*
|
|
|
|
* @return null|Node Found node (or null if none found)
|
|
|
|
*/
|
|
|
|
public function findFirst($nodes, callable $filter) {
|
|
|
|
if (!is_array($nodes)) {
|
|
|
|
$nodes = [$nodes];
|
|
|
|
}
|
|
|
|
|
|
|
|
$visitor = new FirstFindingVisitor($filter);
|
|
|
|
|
|
|
|
$traverser = new NodeTraverser;
|
|
|
|
$traverser->addVisitor($visitor);
|
|
|
|
$traverser->traverse($nodes);
|
|
|
|
|
|
|
|
return $visitor->getFoundNode();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Find first node that is an instance of a certain class.
|
|
|
|
*
|
|
|
|
* @param Node|Node[] $nodes Single node or array of nodes to search in
|
|
|
|
* @param string $class Class name
|
|
|
|
*
|
|
|
|
* @return null|Node Found node, which is an instance of $class (or null if none found)
|
|
|
|
*/
|
2017-04-28 21:40:59 +02:00
|
|
|
public function findFirstInstanceOf($nodes, string $class) {
|
2017-01-29 23:20:53 +01:00
|
|
|
return $this->findFirst($nodes, function ($node) use ($class) {
|
|
|
|
return $node instanceof $class;
|
|
|
|
});
|
|
|
|
}
|
2018-01-10 18:04:06 +01:00
|
|
|
}
|