2010-03-31 07:42:18 +01:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/*
|
2010-04-07 01:51:29 +01:00
|
|
|
* This file is part of the Symfony package.
|
2010-03-31 07:42:18 +01:00
|
|
|
*
|
2011-03-06 11:40:06 +00:00
|
|
|
* (c) Fabien Potencier <fabien@symfony.com>
|
2010-03-31 07:42:18 +01:00
|
|
|
*
|
|
|
|
* For the full copyright and license information, please view the LICENSE
|
|
|
|
* file that was distributed with this source code.
|
|
|
|
*/
|
|
|
|
|
2011-01-15 13:29:43 +00:00
|
|
|
namespace Symfony\Component\CssSelector\Node;
|
|
|
|
|
2011-06-14 13:12:03 +01:00
|
|
|
use Symfony\Component\CssSelector\Exception\ParseException;
|
2011-01-15 13:29:43 +00:00
|
|
|
|
2010-03-31 07:42:18 +01:00
|
|
|
/**
|
|
|
|
* PseudoNode represents a "selector:ident" node.
|
|
|
|
*
|
|
|
|
* This component is a port of the Python lxml library,
|
|
|
|
* which is copyright Infrae and distributed under the BSD license.
|
|
|
|
*
|
2011-03-06 11:40:06 +00:00
|
|
|
* @author Fabien Potencier <fabien@symfony.com>
|
2010-03-31 07:42:18 +01:00
|
|
|
*/
|
|
|
|
class PseudoNode implements NodeInterface
|
|
|
|
{
|
2010-05-06 12:25:53 +01:00
|
|
|
static protected $unsupported = array(
|
|
|
|
'indeterminate', 'first-line', 'first-letter',
|
|
|
|
'selection', 'before', 'after', 'link', 'visited',
|
|
|
|
'active', 'focus', 'hover',
|
|
|
|
);
|
|
|
|
|
|
|
|
protected $element;
|
|
|
|
protected $type;
|
|
|
|
protected $ident;
|
|
|
|
|
|
|
|
/**
|
2011-02-09 00:19:47 +00:00
|
|
|
* Constructor.
|
|
|
|
*
|
|
|
|
* @param NodeInterface $element The NodeInterface element
|
|
|
|
* @param string $type Node type
|
|
|
|
* @param string $ident The ident
|
2011-06-14 13:12:03 +01:00
|
|
|
* @throws ParseException When incorrect PseudoNode type is given
|
2010-05-06 12:25:53 +01:00
|
|
|
*/
|
|
|
|
public function __construct($element, $type, $ident)
|
2010-03-31 07:42:18 +01:00
|
|
|
{
|
2010-05-06 12:25:53 +01:00
|
|
|
$this->element = $element;
|
|
|
|
|
2010-05-07 15:09:11 +01:00
|
|
|
if (!in_array($type, array(':', '::'))) {
|
2011-06-14 13:12:03 +01:00
|
|
|
throw new ParseException(sprintf('The PseudoNode type can only be : or :: (%s given).', $type));
|
2010-05-06 12:25:53 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
$this->type = $type;
|
|
|
|
$this->ident = $ident;
|
|
|
|
}
|
|
|
|
|
2011-02-09 00:19:47 +00:00
|
|
|
/**
|
|
|
|
* {@inheritDoc}
|
|
|
|
*/
|
2010-05-06 12:25:53 +01:00
|
|
|
public function __toString()
|
|
|
|
{
|
|
|
|
return sprintf('%s[%s%s%s]', __CLASS__, $this->element, $this->type, $this->ident);
|
2010-03-31 07:42:18 +01:00
|
|
|
}
|
|
|
|
|
2010-05-06 12:25:53 +01:00
|
|
|
/**
|
2011-02-09 00:19:47 +00:00
|
|
|
* {@inheritDoc}
|
2011-06-14 13:12:03 +01:00
|
|
|
* @throws ParseException When unsupported or unknown pseudo-class is found
|
2010-05-06 12:25:53 +01:00
|
|
|
*/
|
|
|
|
public function toXpath()
|
|
|
|
{
|
2011-02-27 19:56:29 +00:00
|
|
|
$elXpath = $this->element->toXpath();
|
2010-05-06 12:25:53 +01:00
|
|
|
|
2010-05-07 15:09:11 +01:00
|
|
|
if (in_array($this->ident, self::$unsupported)) {
|
2011-06-14 13:12:03 +01:00
|
|
|
throw new ParseException(sprintf('The pseudo-class %s is unsupported', $this->ident));
|
2010-05-06 12:25:53 +01:00
|
|
|
}
|
|
|
|
$method = 'xpath_'.str_replace('-', '_', $this->ident);
|
2010-05-07 15:09:11 +01:00
|
|
|
if (!method_exists($this, $method)) {
|
2011-06-14 13:12:03 +01:00
|
|
|
throw new ParseException(sprintf('The pseudo-class %s is unknown', $this->ident));
|
2010-05-06 12:25:53 +01:00
|
|
|
}
|
|
|
|
|
2011-02-27 19:56:29 +00:00
|
|
|
return $this->$method($elXpath);
|
2010-05-06 12:25:53 +01:00
|
|
|
}
|
2010-03-31 07:42:18 +01:00
|
|
|
|
2011-02-09 00:19:47 +00:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param XPathExpr $xpath The XPath expression
|
|
|
|
* @return XPathExpr The modified XPath expression
|
|
|
|
*/
|
2010-05-06 12:25:53 +01:00
|
|
|
protected function xpath_checked($xpath)
|
|
|
|
{
|
|
|
|
// FIXME: is this really all the elements?
|
|
|
|
$xpath->addCondition("(@selected or @checked) and (name(.) = 'input' or name(.) = 'option')");
|
2010-03-31 07:42:18 +01:00
|
|
|
|
2010-05-06 12:25:53 +01:00
|
|
|
return $xpath;
|
|
|
|
}
|
2010-03-31 07:42:18 +01:00
|
|
|
|
2010-05-06 12:25:53 +01:00
|
|
|
/**
|
2011-02-09 00:19:47 +00:00
|
|
|
* @param XPathExpr $xpath The XPath expression
|
|
|
|
* @return XPathExpr The modified XPath expression
|
2011-06-14 13:12:03 +01:00
|
|
|
* @throws ParseException If this element is the root element
|
2010-05-06 12:25:53 +01:00
|
|
|
*/
|
|
|
|
protected function xpath_root($xpath)
|
2010-03-31 07:42:18 +01:00
|
|
|
{
|
2010-05-06 12:25:53 +01:00
|
|
|
// if this element is the root element
|
2011-06-14 13:12:03 +01:00
|
|
|
throw new ParseException();
|
2010-03-31 07:42:18 +01:00
|
|
|
}
|
2010-05-06 12:25:53 +01:00
|
|
|
|
2011-02-09 00:19:47 +00:00
|
|
|
/**
|
|
|
|
* Marks this XPath expression as the first child.
|
|
|
|
*
|
|
|
|
* @param XPathExpr $xpath The XPath expression
|
|
|
|
* @return XPathExpr The modified expression
|
|
|
|
*/
|
2010-05-06 12:25:53 +01:00
|
|
|
protected function xpath_first_child($xpath)
|
2010-03-31 07:42:18 +01:00
|
|
|
{
|
2010-05-06 12:25:53 +01:00
|
|
|
$xpath->addStarPrefix();
|
|
|
|
$xpath->addNameTest();
|
|
|
|
$xpath->addCondition('position() = 1');
|
|
|
|
|
|
|
|
return $xpath;
|
2010-03-31 07:42:18 +01:00
|
|
|
}
|
|
|
|
|
2011-04-15 20:12:02 +01:00
|
|
|
/**
|
2011-02-09 00:19:47 +00:00
|
|
|
* Sets the XPath to be the last child.
|
|
|
|
*
|
|
|
|
* @param XPathExpr $xpath The XPath expression
|
|
|
|
* @return XPathExpr The modified expression
|
|
|
|
*/
|
2010-05-06 12:25:53 +01:00
|
|
|
protected function xpath_last_child($xpath)
|
2010-03-31 07:42:18 +01:00
|
|
|
{
|
2010-05-06 12:25:53 +01:00
|
|
|
$xpath->addStarPrefix();
|
|
|
|
$xpath->addNameTest();
|
|
|
|
$xpath->addCondition('position() = last()');
|
|
|
|
|
|
|
|
return $xpath;
|
2010-03-31 07:42:18 +01:00
|
|
|
}
|
2010-05-06 12:25:53 +01:00
|
|
|
|
2011-02-09 00:19:47 +00:00
|
|
|
/**
|
|
|
|
* Sets the XPath expression to be the first of type.
|
|
|
|
*
|
|
|
|
* @param XPathExpr $xpath The XPath expression
|
|
|
|
* @return XPathExpr The modified expression
|
|
|
|
*/
|
2010-05-06 12:25:53 +01:00
|
|
|
protected function xpath_first_of_type($xpath)
|
2010-03-31 07:42:18 +01:00
|
|
|
{
|
2010-05-07 15:09:11 +01:00
|
|
|
if ($xpath->getElement() == '*') {
|
2011-06-14 13:12:03 +01:00
|
|
|
throw new ParseException('*:first-of-type is not implemented');
|
2010-05-06 12:25:53 +01:00
|
|
|
}
|
|
|
|
$xpath->addStarPrefix();
|
|
|
|
$xpath->addCondition('position() = 1');
|
|
|
|
|
|
|
|
return $xpath;
|
2010-03-31 07:42:18 +01:00
|
|
|
}
|
2010-05-06 12:25:53 +01:00
|
|
|
|
|
|
|
/**
|
2011-02-09 00:19:47 +00:00
|
|
|
* Sets the XPath expression to be the last of type.
|
|
|
|
*
|
|
|
|
* @param XPathExpr $xpath The XPath expression
|
|
|
|
* @return XPathExpr The modified expression
|
2011-06-14 13:12:03 +01:00
|
|
|
* @throws ParseException Because *:last-of-type is not implemented
|
2010-05-06 12:25:53 +01:00
|
|
|
*/
|
|
|
|
protected function xpath_last_of_type($xpath)
|
2010-03-31 07:42:18 +01:00
|
|
|
{
|
2010-05-07 15:09:11 +01:00
|
|
|
if ($xpath->getElement() == '*') {
|
2011-06-14 13:12:03 +01:00
|
|
|
throw new ParseException('*:last-of-type is not implemented');
|
2010-05-06 12:25:53 +01:00
|
|
|
}
|
|
|
|
$xpath->addStarPrefix();
|
|
|
|
$xpath->addCondition('position() = last()');
|
|
|
|
|
|
|
|
return $xpath;
|
2010-03-31 07:42:18 +01:00
|
|
|
}
|
|
|
|
|
2011-02-09 00:19:47 +00:00
|
|
|
/**
|
|
|
|
* Sets the XPath expression to be the only child.
|
|
|
|
*
|
|
|
|
* @param XPathExpr $xpath The XPath expression
|
|
|
|
* @return XPathExpr The modified expression
|
|
|
|
*/
|
2010-05-06 12:25:53 +01:00
|
|
|
protected function xpath_only_child($xpath)
|
|
|
|
{
|
|
|
|
$xpath->addNameTest();
|
|
|
|
$xpath->addStarPrefix();
|
|
|
|
$xpath->addCondition('last() = 1');
|
|
|
|
|
|
|
|
return $xpath;
|
|
|
|
}
|
2010-03-31 07:42:18 +01:00
|
|
|
|
2010-05-06 12:25:53 +01:00
|
|
|
/**
|
2011-02-09 00:19:47 +00:00
|
|
|
* Sets the XPath expression to be only of type.
|
|
|
|
*
|
|
|
|
* @param XPathExpr $xpath The XPath expression
|
|
|
|
* @return XPathExpr The modified expression
|
2011-06-14 13:12:03 +01:00
|
|
|
* @throws ParseException Because *:only-of-type is not implemented
|
2010-05-06 12:25:53 +01:00
|
|
|
*/
|
|
|
|
protected function xpath_only_of_type($xpath)
|
|
|
|
{
|
2010-05-07 15:09:11 +01:00
|
|
|
if ($xpath->getElement() == '*') {
|
2011-06-14 13:12:03 +01:00
|
|
|
throw new ParseException('*:only-of-type is not implemented');
|
2010-05-06 12:25:53 +01:00
|
|
|
}
|
|
|
|
$xpath->addCondition('last() = 1');
|
2010-03-31 07:42:18 +01:00
|
|
|
|
2010-05-06 12:25:53 +01:00
|
|
|
return $xpath;
|
|
|
|
}
|
|
|
|
|
2011-02-09 00:19:47 +00:00
|
|
|
/**
|
|
|
|
* undocumented function
|
|
|
|
*
|
|
|
|
* @param XPathExpr $xpath The XPath expression
|
|
|
|
* @return XPathExpr The modified expression
|
|
|
|
*/
|
2010-05-06 12:25:53 +01:00
|
|
|
protected function xpath_empty($xpath)
|
|
|
|
{
|
|
|
|
$xpath->addCondition('not(*) and not(normalize-space())');
|
|
|
|
|
|
|
|
return $xpath;
|
|
|
|
}
|
2010-03-31 07:42:18 +01:00
|
|
|
}
|