Modern DOM library written in PHP for HTML documents
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

142 lines
5.6 KiB

<?php
/**
* @license MIT
* Copyright 2017 Dustin Wilson, J. King, et al.
* See LICENSE and AUTHORS files for details
*/
declare(strict_types=1);
namespace MensBeam\HTML\DOM;
use MensBeam\HTML\DOM\Inner\Document as InnerDocument,
MensBeam\HTML\Parser\NameCoercion;
class NamedNodeMap extends Collection {
use NameCoercion;
# A NamedNodeMap has an associated element (an element).
protected Element $element;
protected function __construct(Element $element, InnerDocument $innerDocument, ?\DOMNamedNodeMap $namedNodeMap) {
$this->element = $element;
$this->innerDocument = $innerDocument;
// Have to check for null because PHP DOM violates the spec and returns null when empty
$this->innerCollection = $namedNodeMap ?? new \DOMNamedNodeMap();
}
public function current(): ?Attr {
return parent::current();
}
public function getNamedItem(string $qualifiedName): ?Attr {
# The getNamedItem(qualifiedName) method steps are to return the result of
# getting an attribute given qualifiedName and element.
return $this->element->getAttributeNode($qualifiedName);
}
public function getNamedItemNS(?string $namespace, string $localName): ?Attr {
# The getNamedItemNS(namespace, localName) method steps are to return the result
# of getting an attribute given namespace, localName, and element.
return $this->element->getAttributeNodeNS($namespace, $localName);
}
public function item(int $index): ?Attr {
return parent::item($index);
}
public function offsetGet($offset): ?Attr {
if (is_int($offset)) {
return $this->item($offset);
}
# A NamedNodeMap object’s supported property names are the return value of running
# these steps:
// The spec is extremely vague as to what to do here, but it seems to expect
// this to be some sort of live private property that the class will poll to
// check for valid property names when trying to access them. This is
// inefficient. Going to do basically the same thing but not return a list of
// every one. It will just search the element's attribute list instead using the
// same process.
# 1. Let names be the qualified names of the attributes in this NamedNodeMap object’s
# attribute list, with duplicates omitted, in order.
# 2. If this NamedNodeMap object’s element is in the HTML namespace and its node
# document is an HTML document, then for each name in names:
# 1. Let lowercaseName be name, in ASCII lowercase.
# 2. If lowercaseName is not equal to name, remove name from names.
# 3. Return names.
$innerElement = $this->element->innerNode;
$innerDocument = $innerElement->ownerDocument;
$attributes = $innerElement->attributes;
if ($attributes->length > 0) {
if (strpos(needle: ':', haystack: $offset) !== false) {
$coercedOffset = implode(':', array_map([$this, 'coerceName'], explode(':', $offset, 2)));
} else {
$coercedOffset = $this->coerceName($offset);
}
$document = $this->element->ownerDocument;
foreach ($attributes as $attr) {
$name = $attr->nodeName;
// Coercion causes uppercase characters to be used, causing this part of the
// specification's instructions to fail. Circumvent this by checking for the
// presence of a coerced name.
if ($this->element->namespaceURI === Node::HTML_NAMESPACE && !$document instanceof XMLDocument && !preg_match('/U[0-9A-F]{6}/', $name) && $name !== strtolower($name)) {
// This can only be reached if an attribute is created on an XML document,
// imported into an HTML document, and then appended to an element.
continue;
}
if ($name === $offset || $name === $coercedOffset) {
return $innerDocument->getWrapperNode($attr);
}
}
}
return null;
}
public function offsetExists($offset): bool {
return ($this->offsetGet($offset) !== null);
}
public function removeNamedItem(string $qualifiedName): ?Attr {
return $this->removeNamedItemNS(null, $qualifiedName);
}
public function removeNamedItemNS(?string $namespace, string $localName): ?Attr {
# The removeNamedItem(qualifiedName) method steps are:
#
# 1. Let attr be the result of removing an attribute given namespace, localName,
# and element.
// Going to get the attribute node instead here. The result at the end will be
// the same.
$attr = $this->element->getAttributeNodeNS($namespace, $localName);
# 2. If attr is null, then throw a "NotFoundError" DOMException.
if ($attr === null) {
throw new NotFoundError();
}
# 3. Return attr.
return $this->element->removeAttributeNode($attr);
}
public function setNamedItem(Attr $attr): ?Attr {
# The setNamedItem(attr) and setNamedItemNS(attr) method steps are to return the
# result of setting an attribute given attr and element.
return $this->element->setAttributeNode($attr);
}
public function setNamedItemNS(Attr $attr): ?Attr {
# The setNamedItem(attr) and setNamedItemNS(attr) method steps are to return the
# result of setting an attribute given attr and element.
return $this->element->setAttributeNode($attr);
}
}