1: | <?php declare(strict_types=1); |
2: | |
3: | namespace PhpParser\NodeVisitor; |
4: | |
5: | use PhpParser\Comment; |
6: | use PhpParser\Node; |
7: | use PhpParser\NodeVisitorAbstract; |
8: | use PhpParser\Token; |
9: | |
10: | class CommentAnnotatingVisitor extends NodeVisitorAbstract { |
11: | |
12: | private int $pos = 0; |
13: | |
14: | private array $tokens; |
15: | |
16: | private array $commentPositions = []; |
17: | |
18: | |
19: | |
20: | |
21: | |
22: | |
23: | public function __construct(array $tokens) { |
24: | $this->tokens = $tokens; |
25: | |
26: | |
27: | |
28: | foreach ($tokens as $i => $token) { |
29: | if ($token->id === \T_COMMENT || $token->id === \T_DOC_COMMENT) { |
30: | $this->commentPositions[] = $i; |
31: | } |
32: | } |
33: | } |
34: | |
35: | public function enterNode(Node $node) { |
36: | $nextCommentPos = current($this->commentPositions); |
37: | if ($nextCommentPos === false) { |
38: | |
39: | return self::STOP_TRAVERSAL; |
40: | } |
41: | |
42: | $oldPos = $this->pos; |
43: | $this->pos = $pos = $node->getStartTokenPos(); |
44: | if ($nextCommentPos > $oldPos && $nextCommentPos < $pos) { |
45: | $comments = []; |
46: | while (--$pos >= $oldPos) { |
47: | $token = $this->tokens[$pos]; |
48: | if ($token->id === \T_DOC_COMMENT) { |
49: | $comments[] = new Comment\Doc( |
50: | $token->text, $token->line, $token->pos, $pos, |
51: | $token->getEndLine(), $token->getEndPos() - 1, $pos); |
52: | continue; |
53: | } |
54: | if ($token->id === \T_COMMENT) { |
55: | $comments[] = new Comment( |
56: | $token->text, $token->line, $token->pos, $pos, |
57: | $token->getEndLine(), $token->getEndPos() - 1, $pos); |
58: | continue; |
59: | } |
60: | if ($token->id !== \T_WHITESPACE) { |
61: | break; |
62: | } |
63: | } |
64: | if (!empty($comments)) { |
65: | $node->setAttribute('comments', array_reverse($comments)); |
66: | } |
67: | |
68: | do { |
69: | $nextCommentPos = next($this->commentPositions); |
70: | } while ($nextCommentPos !== false && $nextCommentPos < $this->pos); |
71: | } |
72: | |
73: | $endPos = $node->getEndTokenPos(); |
74: | if ($nextCommentPos > $endPos) { |
75: | |
76: | $this->pos = $endPos; |
77: | return self::DONT_TRAVERSE_CHILDREN; |
78: | } |
79: | |
80: | return null; |
81: | } |
82: | } |
83: | |