Вход Регистрация
Файл: vendor/symfony/css-selector/Parser/Parser.php
Строк: 372
<?php

/*
 * This file is part of the Symfony package.
 *
 * (c) Fabien Potencier <fabien@symfony.com>
 *
 * For the full copyright and license information, please view the LICENSE
 * file that was distributed with this source code.
 */

namespace SymfonyComponentCssSelectorParser;

use 
SymfonyComponentCssSelectorExceptionSyntaxErrorException;
use 
SymfonyComponentCssSelectorNode;
use 
SymfonyComponentCssSelectorParserTokenizerTokenizer;

/**
 * CSS selector parser.
 *
 * This component is a port of the Python cssselect library,
 * which is copyright Ian Bicking, @see https://github.com/scrapy/cssselect.
 *
 * @author Jean-François Simon <jeanfrancois.simon@sensiolabs.com>
 *
 * @internal
 */
class Parser implements ParserInterface
{
    private 
Tokenizer $tokenizer;

    public function 
__construct(?Tokenizer $tokenizer null)
    {
        
$this->tokenizer $tokenizer ?? new Tokenizer();
    }

    public function 
parse(string $source): array
    {
        
$reader = new Reader($source);
        
$stream $this->tokenizer->tokenize($reader);

        return 
$this->parseSelectorList($stream);
    }

    
/**
     * Parses the arguments for ":nth-child()" and friends.
     *
     * @param Token[] $tokens
     *
     * @throws SyntaxErrorException
     */
    
public static function parseSeries(array $tokens): array
    {
        foreach (
$tokens as $token) {
            if (
$token->isString()) {
                throw 
SyntaxErrorException::stringAsFunctionArgument();
            }
        }

        
$joined trim(implode(''array_map(fn (Token $token) => $token->getValue(), $tokens)));

        
$int = function ($string) {
            if (!
is_numeric($string)) {
                throw 
SyntaxErrorException::stringAsFunctionArgument();
            }

            return (int) 
$string;
        };

        switch (
true) {
            case 
'odd' === $joined:
                return [
21];
            case 
'even' === $joined:
                return [
20];
            case 
'n' === $joined:
                return [
10];
            case !
str_contains($joined'n'):
                return [
0$int($joined)];
        }

        
$split explode('n'$joined);
        
$first $split[0] ?? null;

        return [
            
$first ? ('-' === $first || '+' === $first $int($first.'1') : $int($first)) : 1,
            isset(
$split[1]) && $split[1] ? $int($split[1]) : 0,
        ];
    }

    private function 
parseSelectorList(TokenStream $stream): array
    {
        
$stream->skipWhitespace();
        
$selectors = [];

        while (
true) {
            
$selectors[] = $this->parserSelectorNode($stream);

            if (
$stream->getPeek()->isDelimiter([','])) {
                
$stream->getNext();
                
$stream->skipWhitespace();
            } else {
                break;
            }
        }

        return 
$selectors;
    }

    private function 
parserSelectorNode(TokenStream $stream): NodeSelectorNode
    
{
        [
$result$pseudoElement] = $this->parseSimpleSelector($stream);

        while (
true) {
            
$stream->skipWhitespace();
            
$peek $stream->getPeek();

            if (
$peek->isFileEnd() || $peek->isDelimiter([','])) {
                break;
            }

            if (
null !== $pseudoElement) {
                throw 
SyntaxErrorException::pseudoElementFound($pseudoElement'not at the end of a selector');
            }

            if (
$peek->isDelimiter(['+''>''~'])) {
                
$combinator $stream->getNext()->getValue();
                
$stream->skipWhitespace();
            } else {
                
$combinator ' ';
            }

            [
$nextSelector$pseudoElement] = $this->parseSimpleSelector($stream);
            
$result = new NodeCombinedSelectorNode($result$combinator$nextSelector);
        }

        return new 
NodeSelectorNode($result$pseudoElement);
    }

    
/**
     * Parses next simple node (hash, class, pseudo, negation).
     *
     * @throws SyntaxErrorException
     */
    
private function parseSimpleSelector(TokenStream $streambool $insideNegation false): array
    {
        
$stream->skipWhitespace();

        
$selectorStart count($stream->getUsed());
        
$result $this->parseElementNode($stream);
        
$pseudoElement null;

        while (
true) {
            
$peek $stream->getPeek();
            if (
$peek->isWhitespace()
                || 
$peek->isFileEnd()
                || 
$peek->isDelimiter([',''+''>''~'])
                || (
$insideNegation && $peek->isDelimiter([')']))
            ) {
                break;
            }

            if (
null !== $pseudoElement) {
                throw 
SyntaxErrorException::pseudoElementFound($pseudoElement'not at the end of a selector');
            }

            if (
$peek->isHash()) {
                
$result = new NodeHashNode($result$stream->getNext()->getValue());
            } elseif (
$peek->isDelimiter(['.'])) {
                
$stream->getNext();
                
$result = new NodeClassNode($result$stream->getNextIdentifier());
            } elseif (
$peek->isDelimiter(['['])) {
                
$stream->getNext();
                
$result $this->parseAttributeNode($result$stream);
            } elseif (
$peek->isDelimiter([':'])) {
                
$stream->getNext();

                if (
$stream->getPeek()->isDelimiter([':'])) {
                    
$stream->getNext();
                    
$pseudoElement $stream->getNextIdentifier();

                    continue;
                }

                
$identifier $stream->getNextIdentifier();
                if (
in_array(strtolower($identifier), ['first-line''first-letter''before''after'])) {
                    
// Special case: CSS 2.1 pseudo-elements can have a single ':'.
                    // Any new pseudo-element must have two.
                    
$pseudoElement $identifier;

                    continue;
                }

                if (!
$stream->getPeek()->isDelimiter(['('])) {
                    
$result = new NodePseudoNode($result$identifier);
                    if (
'Pseudo[Element[*]:scope]' === $result->__toString()) {
                        
$used count($stream->getUsed());
                        if (!(
=== $used
                           
|| === $used && $stream->getUsed()[0]->isWhiteSpace()
                           || 
$used >= && $stream->getUsed()[$used 3]->isDelimiter([','])
                           || 
$used >= 4
                                
&& $stream->getUsed()[$used 3]->isWhiteSpace()
                                && 
$stream->getUsed()[$used 4]->isDelimiter([','])
                        )) {
                            throw 
SyntaxErrorException::notAtTheStartOfASelector('scope');
                        }
                    }
                    continue;
                }

                
$stream->getNext();
                
$stream->skipWhitespace();

                if (
'not' === strtolower($identifier)) {
                    if (
$insideNegation) {
                        throw 
SyntaxErrorException::nestedNot();
                    }

                    [
$argument$argumentPseudoElement] = $this->parseSimpleSelector($streamtrue);
                    
$next $stream->getNext();

                    if (
null !== $argumentPseudoElement) {
                        throw 
SyntaxErrorException::pseudoElementFound($argumentPseudoElement'inside ::not()');
                    }

                    if (!
$next->isDelimiter([')'])) {
                        throw 
SyntaxErrorException::unexpectedToken('")"'$next);
                    }

                    
$result = new NodeNegationNode($result$argument);
                } else {
                    
$arguments = [];
                    
$next null;

                    while (
true) {
                        
$stream->skipWhitespace();
                        
$next $stream->getNext();

                        if (
$next->isIdentifier()
                            || 
$next->isString()
                            || 
$next->isNumber()
                            || 
$next->isDelimiter(['+''-'])
                        ) {
                            
$arguments[] = $next;
                        } elseif (
$next->isDelimiter([')'])) {
                            break;
                        } else {
                            throw 
SyntaxErrorException::unexpectedToken('an argument'$next);
                        }
                    }

                    if (!
$arguments) {
                        throw 
SyntaxErrorException::unexpectedToken('at least one argument'$next);
                    }

                    
$result = new NodeFunctionNode($result$identifier$arguments);
                }
            } else {
                throw 
SyntaxErrorException::unexpectedToken('selector'$peek);
            }
        }

        if (
count($stream->getUsed()) === $selectorStart) {
            throw 
SyntaxErrorException::unexpectedToken('selector'$stream->getPeek());
        }

        return [
$result$pseudoElement];
    }

    private function 
parseElementNode(TokenStream $stream): NodeElementNode
    
{
        
$peek $stream->getPeek();

        if (
$peek->isIdentifier() || $peek->isDelimiter(['*'])) {
            if (
$peek->isIdentifier()) {
                
$namespace $stream->getNext()->getValue();
            } else {
                
$stream->getNext();
                
$namespace null;
            }

            if (
$stream->getPeek()->isDelimiter(['|'])) {
                
$stream->getNext();
                
$element $stream->getNextIdentifierOrStar();
            } else {
                
$element $namespace;
                
$namespace null;
            }
        } else {
            
$element $namespace null;
        }

        return new 
NodeElementNode($namespace$element);
    }

    private function 
parseAttributeNode(NodeNodeInterface $selectorTokenStream $stream): NodeAttributeNode
    
{
        
$stream->skipWhitespace();
        
$attribute $stream->getNextIdentifierOrStar();

        if (
null === $attribute && !$stream->getPeek()->isDelimiter(['|'])) {
            throw 
SyntaxErrorException::unexpectedToken('"|"'$stream->getPeek());
        }

        if (
$stream->getPeek()->isDelimiter(['|'])) {
            
$stream->getNext();

            if (
$stream->getPeek()->isDelimiter(['='])) {
                
$namespace null;
                
$stream->getNext();
                
$operator '|=';
            } else {
                
$namespace $attribute;
                
$attribute $stream->getNextIdentifier();
                
$operator null;
            }
        } else {
            
$namespace $operator null;
        }

        if (
null === $operator) {
            
$stream->skipWhitespace();
            
$next $stream->getNext();

            if (
$next->isDelimiter([']'])) {
                return new 
NodeAttributeNode($selector$namespace$attribute'exists'null);
            } elseif (
$next->isDelimiter(['='])) {
                
$operator '=';
            } elseif (
$next->isDelimiter(['^''$''*''~''|''!'])
                && 
$stream->getPeek()->isDelimiter(['='])
            ) {
                
$operator $next->getValue().'=';
                
$stream->getNext();
            } else {
                throw 
SyntaxErrorException::unexpectedToken('operator'$next);
            }
        }

        
$stream->skipWhitespace();
        
$value $stream->getNext();

        if (
$value->isNumber()) {
            
// if the value is a number, it's casted into a string
            
$value = new Token(Token::TYPE_STRING, (string) $value->getValue(), $value->getPosition());
        }

        if (!(
$value->isIdentifier() || $value->isString())) {
            throw 
SyntaxErrorException::unexpectedToken('string or identifier'$value);
        }

        
$stream->skipWhitespace();
        
$next $stream->getNext();

        if (!
$next->isDelimiter([']'])) {
            throw 
SyntaxErrorException::unexpectedToken('"]"'$next);
        }

        return new 
NodeAttributeNode($selector$namespace$attribute$operator$value->getValue());
    }
}
Онлайн: 1
Реклама