Browse Source

Added Attr, HTMLCollection

wrapper-classes
Dustin Wilson 3 years ago
parent
commit
cc3056da45
  1. 52
      lib/Attr.php
  2. 6
      lib/Comment.php
  3. 9
      lib/Document.php
  4. 4
      lib/DocumentFragment.php
  5. 3
      lib/Element.php
  6. 166
      lib/HTMLCollection.php
  7. 2
      lib/Node.php
  8. 2
      lib/NodeList.php
  9. 6
      lib/ProcessingInstruction.php
  10. 17
      lib/Text.php

52
lib/Attr.php

@ -0,0 +1,52 @@
<?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;
class Attr extends Node {
protected function __get_localName(): string {
// PHP's DOM does this correctly already.
return $this->innerNode->localName;
}
protected function __get_name(): string {
// PHP's DOM does this correctly already.
return $this->innerNode->name;
}
protected function __get_namespaceURI(): string {
// PHP's DOM does this correctly already.
return $this->innerNode->namespaceURI;
}
protected function __get_ownerElement(): Element {
// PHP's DOM does this correctly already.
return $this->innerNode->ownerDocument->getWrapperNode($this->innerNode->ownerElement);
}
protected function __get_prefix(): string {
// PHP's DOM does this correctly already.
return $this->innerNode->prefix;
}
protected function __get_specified(): bool {
# Useless, always returns true
return true;
}
protected function __get_value(): string {
// PHP's DOM does this correctly already.
return $this->innerNode->value;
}
protected function __set_value(string $value) {
// PHP's DOM does this correctly already.
$this->innerNode->value = $value;
}
}

6
lib/Comment.php

@ -9,4 +9,8 @@ declare(strict_types=1);
namespace MensBeam\HTML\DOM;
class Comment extends CharacterData {}
class Comment extends CharacterData {
public function __construct(string $data = '') {
$this->innerNode = new \DOMComment($data);
}
}

9
lib/Document.php

@ -36,7 +36,14 @@ class Document extends Node {
public function createDocumentFragment(): DocumentFragment {
return $this->innerNode->getWrapperNode($this->innerNode->createDocumentFragment());
// DocumentFragment has a public constructor that creates an inner fragment
// without an associated document, so some jiggerypokery must be done instead.
$reflector = new \ReflectionClass(__NAMESPACE__ . '\\DocumentFragment');
$fragment = $reflector->newInstanceWithoutConstructor();
$property = new \ReflectionProperty($text, 'innerNode');
$property->setAccessible(true);
$property->setValue($fragment, $this->innerNode->createDocumentFragment());
return $fragment;
}
public function createElement(string $localName): Element {

4
lib/DocumentFragment.php

@ -15,7 +15,7 @@ class DocumentFragment extends Node {
protected ?\WeakReference $host = null;
protected function __construct(\DOMDocumentFragment $fragment) {
parent::__construct($fragment);
public function __construct() {
$this->innerNode = new \DOMDocumentFragment();
}
}

3
lib/Element.php

@ -19,7 +19,8 @@ class Element extends Node {
// around because of the wrapper classes; So, use the null namespace internally
// but print out the HTML namespace instead.
$namespace = $this->innerNode->namespaceURI;
return ($namespace === null) ? Parser::HTML_NAMESPACE : $namespace;
$doc = $this->ownerDocument;
return (($doc instanceof Document && !$doc instanceof XMLDocument) && $namespace === null) ? Parser::HTML_NAMESPACE : $namespace;
}

166
lib/HTMLCollection.php

@ -0,0 +1,166 @@
<?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\Framework\MagicProperties,
MensBeam\HTML\Parser;
# An HTMLCollection object is a collection of elements.
#
# NOTE: HTMLCollection is a historical artifact we cannot rid the web of. While
# developers are of course welcome to keep using it, new API standard designers
# ought not to use it (use sequence<T> in IDL instead).
#
# A collection is an object that represents a list of nodes. A collection can be
# either live or static. Unless otherwise stated, a collection must be live.
#
# If a collection is live, then the attributes and methods on that object must
# operate on the actual underlying data, not a snapshot of the data.
#
# When a collection is created, a filter and a root are associated with it.
#
# The collection then represents a view of the subtree rooted at the
# collection’s root, containing only nodes that match the given filter. The view
# is linear. In the absence of specific requirements to the contrary, the nodes
# within the collection must be sorted in tree order.
class HTMLCollection implements \ArrayAccess, \Countable, \Iterator {
use MagicProperties;
protected ?\Closure $filter = null;
protected int $_length = 0;
protected ?array $nodeArray = null;
protected int $position = 0;
protected function __get_length(): int {
# The length attribute must return the number of nodes represented by the
# collection.
return $this->count();
}
protected function __construct(array|\Closure $arrayOrClosure = []) {
// In this implementation the root part of the creation is handled either before
// the NodeList is created (array) or within the filter (\Closure).
if ($arrayOrClosure === null) {
$arrayOrClosure = [];
}
if (is_callable($arrayOrClosure)) {
$this->filter = $arrayOrClosure;
} else {
// Check types while also unpacking the iterable.
$array = [];
foreach ($arrayOrClosure as $i) {
if (!$i instanceof Element) {
$type = gettype($i);
if ($type === 'object') {
$type = get_class($i);
}
throw new Exception(Exception::ARGUMENT_TYPE_ERROR, 1, 'arrayOrClosure', 'array<Element>|\\Closure<array<Element>>', $type);
}
$array[] = $i;
}
$this->nodeArray = $array;
$this->_length = count($array);
}
}
public function count(): int {
if ($this->nodeArray !== null) {
return $this->_length;
}
$nodeArray = ($this->filter)();
return count($nodeArray);
}
public function current(): ?Element {
return $this->item($this->position);
}
public function item(int $index): ?Element {
# The item(index) method must return the indexth node in the collection. If
# there is no indexth node in the collection, then the method must return null.
if ($index >= $this->count()) {
return null;
}
$nodeArray = ($this->nodeArray !== null) ? $this->nodeArray : ($this->filter)();
if (array_key_exists($index, $nodeArray)) {
return $nodeArray[$index];
}
return null;
}
public function key(): int {
return $this->position;
}
public function namedItem(string $name): ?Element {
# The namedItem(key) method steps are:
// The interface says to use "name" as the argument while the steps below show
// to use "key" *shrug*.
# 1. If key is the empty string, return null.
if ($name === '') {
return null;
}
# 2. Return the first element in the collection for which at least one of the following is true:
# • it has an ID which is key;
# • it is in the HTML namespace and has a name attribute whose value is key;
# or null if there is no such element.
$nodeArray = ($this->nodeArray !== null) ? $this->nodeArray : ($this->filter)();
foreach ($nodeArray as $element) {
if ($element->getAttribute('id') === $name || $element->namespaceURI === Parser::HTML_NAMESPACE && $element->getAttribute('name') === $name) {
return $element;
}
}
return null;
}
public function next(): void {
$this->position++;
}
public function rewind(): void {
$this->position = 0;
}
public function offsetExists($offset): bool {
if (is_int($offset)) {
$nodeArray = ($this->nodeArray !== null) ? $this->nodeArray : ($this->filter)();
return array_key_exists($offset, $nodeArray);
}
return ($this->namedItem($offset) !== null);
}
public function offsetGet($offset): ?Element {
return (is_int($offset)) ? $this->item($offset) : $this->namedItem($offset);
}
public function offsetSet($offset, $value): void {
// NodeLists are immutable; the spec is ambiguous as to what to do here.
// Browsers silently fail here, so that's what we're going to do.
}
public function offsetUnset($offset): void {
// NodeLists are immutable; the spec is ambiguous as to what to do here.
// Browsers silently fail here, so that's what we're going to do.
}
public function valid() {
$this->offsetExists($this->position);
}
}

2
lib/Node.php

@ -332,7 +332,7 @@ abstract class Node {
})->current() !== null);
}
public function getRootNode(array $options): Node {
public function getRootNode(array $options = []): Node {
# The getRootNode(options) method steps are to return this’s shadow-including
# root if options["composed"] is true; otherwise this’s root.
// DEVIATION: This implementation does not have scripting, so there's no Shadow

2
lib/NodeList.php

@ -58,7 +58,7 @@ class NodeList implements \ArrayAccess, \Countable, \Iterator {
if ($type === 'object') {
$type = get_class($i);
}
throw new DOMException(DOMException::ARGUMENT_TYPE_ERROR, 1, 'arrayOrClosure', 'array<Node>|\\Closure<array<Node>>', $type);
throw new Exception(Exception::ARGUMENT_TYPE_ERROR, 1, 'arrayOrClosure', 'array<Node>|\\Closure<array<Node>>', $type);
}
$array[] = $i;

6
lib/ProcessingInstruction.php

@ -9,4 +9,8 @@ declare(strict_types=1);
namespace MensBeam\HTML\DOM;
class ProcessingInstruction extends CharacterData {}
class ProcessingInstruction extends CharacterData {
protected function __get_target(): string {
return $this->innerNode->target;
}
}

17
lib/Text.php

@ -10,7 +10,24 @@ namespace MensBeam\HTML\DOM;
class Text extends CharacterData {
protected function __get_wholeText(): string {
// PHP's DOM does this correctly already.
return $this->innerNode->wholeText;
}
public function __construct(string $data = '') {
$this->innerNode = new \DOMText($data);
}
public function splitText(int $offset): Text {
// PHP DOM mostly handles this correctly with the exception of not throwing an
// exception when the offset is greater than the length, so let's fix that.
if ($offset > $this->length) {
throw new DOMException(DOMException::INDEX_SIZE_ERROR);
}
return $this->innerNode->ownerDocument->getWrapperNode($this->innerNode->splitText($offset));
}
}
Loading…
Cancel
Save