rector/packages/BetterPhpDocParser/ValueObject/Parser/BetterTokenIterator.php
Tomas Votruba 28c719d1fb Updated Rector to commit 8961d20f56fa4b39bb21feec1f24422398f13dba
8961d20f56 [DX] Localize multi instance checker and privates accessor from Symplify + bump to PHPStan 1.8.3 (#2883)
2022-09-01 19:30:48 +00:00

137 lines
3.7 KiB
PHP

<?php
declare (strict_types=1);
namespace Rector\BetterPhpDocParser\ValueObject\Parser;
use PHPStan\PhpDocParser\Parser\TokenIterator;
use Rector\Core\Exception\ShouldNotHappenException;
use Rector\Core\Util\Reflection\PrivatesAccessor;
final class BetterTokenIterator extends TokenIterator
{
/**
* @var string
*/
private const TOKENS = 'tokens';
/**
* @var string
*/
private const INDEX = 'index';
/**
* @readonly
* @var \Rector\Core\Util\Reflection\PrivatesAccessor
*/
private $privatesAccessor;
/**
* @param array<int, mixed> $tokens
*/
public function __construct(array $tokens, int $index = 0)
{
$this->privatesAccessor = new PrivatesAccessor();
if ($tokens === []) {
$this->privatesAccessor->setPrivateProperty($this, self::TOKENS, []);
$this->privatesAccessor->setPrivateProperty($this, self::INDEX, 0);
} else {
parent::__construct($tokens, $index);
}
}
/**
* @param int[] $types
*/
public function isNextTokenTypes(array $types) : bool
{
foreach ($types as $type) {
if ($this->isNextTokenType($type)) {
return \true;
}
}
return \false;
}
public function isTokenTypeOnPosition(int $tokenType, int $position) : bool
{
$tokens = $this->getTokens();
$token = $tokens[$position] ?? null;
if ($token === null) {
return \false;
}
return $token[1] === $tokenType;
}
public function isNextTokenType(int $tokenType) : bool
{
if ($this->nextTokenType() === null) {
return \false;
}
return $this->nextTokenType() === $tokenType;
}
public function printFromTo(int $from, int $to) : string
{
if ($to < $from) {
throw new ShouldNotHappenException('Arguments are flipped');
}
$tokens = $this->getTokens();
$content = '';
foreach ($tokens as $key => $token) {
if ($key < $from) {
continue;
}
if ($key >= $to) {
continue;
}
$content .= $token[0];
}
return $content;
}
public function print() : string
{
$content = '';
foreach ($this->getTokens() as $token) {
$content .= $token[0];
}
return $content;
}
public function nextTokenType() : ?int
{
$tokens = $this->getTokens();
// does next token exist?
$nextIndex = $this->currentPosition() + 1;
if (!isset($tokens[$nextIndex])) {
return null;
}
$this->pushSavePoint();
$this->next();
$nextTokenType = $this->currentTokenType();
$this->rollback();
return $nextTokenType;
}
public function currentPosition() : int
{
return $this->privatesAccessor->getPrivateProperty($this, self::INDEX);
}
/**
* @return array<array{0: string, 1: int}>
*/
public function getTokens() : array
{
return $this->privatesAccessor->getPrivateProperty($this, self::TOKENS);
}
public function count() : int
{
return \count($this->getTokens());
}
/**
* @return array<array{0: string, 1: int}>
*/
public function partialTokens(int $start, int $end) : array
{
return \array_slice($this->getTokens(), $start, $end);
}
public function containsTokenType(int $type) : bool
{
foreach ($this->getTokens() as $token) {
if ($token[1] === $type) {
return \true;
}
}
return \false;
}
}