1: | <?php declare(strict_types=1); |
2: | |
3: | namespace PhpParser\Lexer\TokenEmulator; |
4: | |
5: | use PhpParser\PhpVersion; |
6: | use PhpParser\Token; |
7: | |
8: | final class NullsafeTokenEmulator extends TokenEmulator { |
9: | public function getPhpVersion(): PhpVersion { |
10: | return PhpVersion::fromComponents(8, 0); |
11: | } |
12: | |
13: | public function isEmulationNeeded(string $code): bool { |
14: | return strpos($code, '?->') !== false; |
15: | } |
16: | |
17: | public function emulate(string $code, array $tokens): array { |
18: | |
19: | |
20: | for ($i = 0, $c = count($tokens); $i < $c; ++$i) { |
21: | $token = $tokens[$i]; |
22: | if ($token->text === '?' && isset($tokens[$i + 1]) && $tokens[$i + 1]->id === \T_OBJECT_OPERATOR) { |
23: | array_splice($tokens, $i, 2, [ |
24: | new Token(\T_NULLSAFE_OBJECT_OPERATOR, '?->', $token->line, $token->pos), |
25: | ]); |
26: | $c--; |
27: | continue; |
28: | } |
29: | |
30: | |
31: | if ($token->id === \T_ENCAPSED_AND_WHITESPACE && isset($tokens[$i - 1]) |
32: | && $tokens[$i - 1]->id === \T_VARIABLE |
33: | && preg_match('/^\?->([a-zA-Z_\x80-\xff][a-zA-Z0-9_\x80-\xff]*)/', $token->text, $matches) |
34: | ) { |
35: | $replacement = [ |
36: | new Token(\T_NULLSAFE_OBJECT_OPERATOR, '?->', $token->line, $token->pos), |
37: | new Token(\T_STRING, $matches[1], $token->line, $token->pos + 3), |
38: | ]; |
39: | $matchLen = \strlen($matches[0]); |
40: | if ($matchLen !== \strlen($token->text)) { |
41: | $replacement[] = new Token( |
42: | \T_ENCAPSED_AND_WHITESPACE, |
43: | \substr($token->text, $matchLen), |
44: | $token->line, $token->pos + $matchLen |
45: | ); |
46: | } |
47: | array_splice($tokens, $i, 1, $replacement); |
48: | $c += \count($replacement) - 1; |
49: | continue; |
50: | } |
51: | } |
52: | |
53: | return $tokens; |
54: | } |
55: | |
56: | public function reverseEmulate(string $code, array $tokens): array { |
57: | |
58: | return $tokens; |
59: | } |
60: | } |
61: | |