1: 2: 3: 4: 5: 6: 7: 8: 9: 10: 11: 12: 13: 14: 15: 16: 17: 18: 19: 20: 21: 22: 23: 24: 25: 26: 27: 28: 29: 30: 31: 32: 33: 34: 35: 36: 37: 38: 39: 40: 41: 42: 43: 44: 45: 46: 47: 48: 49: 50: 51: 52: 53: 54: 55: 56: 57: 58: 59: 60: 61: 62: 63: 64: 65: 66: 67: 68: 69: 70: 71: 72: 73: 74: 75: 76: 77: 78: 79: 80: 81: 82: 83: 84: 85: 86: 87: 88: 89: 90: 91: 92: 93: 94: 95: 96: 97: 98: 99: 100: 101: 102: 103: 104: 105: 106: 107: 108: 109: 110: 111: 112: 113: 114: 115: 116: 117: 118: 119: 120: 121: 122: 123: 124: 125: 126: 127: 128: 129: 130: 131: 132: 133: 134: 135: 136: 137: 138: 139: 140: 141: 142: 143: 144: 145: 146: 147: 148: 149: 150: 151: 152: 153: 154: 155: 156: 157: 158: 159: 160: 161: 162: 163: 164: 165: 166: 167: 168: 169: 170: 171: 172: 173: 174: 175: 176: 177: 178: 179: 180: 181: 182: 183: 184: 185: 186: 187: 188: 189: 190: 191: 192: 193: 194: 195: 196: 197: 198: 199: 200: 201: 202: 203: 204: 205: 206: 207: 208: 209: 210: 211: 212: 213: 214: 215: 216: 217: 218: 219: 220: 221: 222: 223: 224: 225: 226: 227: 228: 229: 230: 231: 232: 233: 234: 235: 236: 237: 238: 239: 240: 241: 242: 243: 244: 245: 246: 247: 248: 249: 250: 251: 252: 253: 254: 255: 256: 257: 258: 259: 260: 261: 262: 263: 264: 265: 266: 267: 268: 269: 270: 271: 272: 273: 274: 275: 276: 277: 278: 279: 280: 281: 282: 283: 284: 285: 286: 287: 288: 289: 290: 291: 292: 293: 294: 295: 296: 297: 298: 299: 300: 301: 302: 303: 304: 305: 306: 307: 308: 309: 310: 311: 312: 313: 314: 315: 316: 317: 318: 319: 320: 321: 322: 323: 324: 325: 326: 327: 328: 329: 330: 331: 332: 333: 334: 335: 336: 337: 338: 339: 340: 341: 342: 343: 344: 345: 346: 347: 348: 349: 350: 351: 352: 353: 354: 355: 356: 357: 358: 359: 360: 361: 362: 363: 364: 365: 366: 367: 368: 369: 370: 371: 372: 373: 374: 375: 376: 377: 378: 379: 380: 381: 382: 383: 384: 385: 386: 387: 388: 389: 390: 391: 392: 393: 394: 395: 396: 397: 398: 399: 400: 401: 402: 403: 404: 405: 406: 407: 408: 409: 410: 411: 412: 413: 414: 415: 416: 417: 418: 419: 420: 421: 422: 423: 424: 425: 426: 427: 428: 429: 430: 431: 432: 433: 434: 435: 436: 437: 438: 439: 440: 441: 442: 443: 444: 445: 446: 447: 448: 449: 450: 451: 452: 453: 454: 455: 456: 457: 458: 459: 460: 461: 462: 463: 464: 465: 466: 467: 468: 469: 470: 471: 472: 473: 474: 475: 476: 477: 478: 479: 480: 481: 482: 483: 484: 485: 486: 487: 488: 489: 490: 491: 492: 493: 494: 495: 496: 497: 498: 499: 500: 501: 502: 503: 504: 505: 506: 507: 508: 509: 510: 511: 512: 513: 514: 515: 516: 517: 518: 519: 520: 521: 522: 523: 524: 525: 526: 527: 528: 529: 530: 531: 532: 533: 534: 535: 536: 537: 538: 539: 540: 541: 542: 543: 544: 545: 546: 547: 548: 549: 550: 551: 552: 553: 554: 555: 556: 557: 558: 559: 560: 561: 562: 563: 564: 565: 566: 567: 568: 569: 570: 571: 572: 573: 574: 575: 576: 577: 578: 579: 580: 581: 582: 583: 584: 585: 586: 587: 588: 589: 590: 591: 592: 593: 594: 595: 596: 597: 598: 599: 600: 601: 602: 603: 604: 605: 606: 607: 608: 609: 610: 611: 612: 613: 614: 615: 616: 617: 618: 619: 620: 621: 622: 623: 624: 625: 626: 627: 628: 629: 630: 631: 632: 633: 634: 635: 636: 637: 638: 639: 640: 641: 642: 643: 644: 645: 646: 647: 648: 649: 650: 651: 652: 653: 654: 655: 656: 657: 658: 659: 660: 661: 662: 663: 664:
<?php
namespace WPGMZA;
require_once(plugin_dir_path(__FILE__) . 'class.selector-to-xpath.php');
class DOMElement extends \DOMElement
{
protected static $xpathConverter;
public function __construct()
{
\DOMElement::__construct();
}
public function querySelector($query)
{
$results = $this->querySelectorAll($query);
if(empty($results))
return null;
return $results[0];
}
public function querySelectorAll($query, $sort=true)
{
$xpath = new \DOMXPath($this->ownerDocument);
try{
$expr = DOMElement::selectorToXPath($query);
}catch(Exception $e) {
echo "<p class='notice notice-warning'>Failed to convert CSS selector to XPath (" . $e->getMessage() . ")</p>";
}
$list = $xpath->query($expr, $this);
$results = array();
for($i = 0; $i < $list->length; $i++)
array_push($results, $list->item($i));
if($sort)
usort($results, array('WPGMZA\DOMElement', 'sortByDOMPosition'));
return $results;
}
public function prepend($subject)
{
if(is_array($subject))
{
$originalFirst = $this->firstChild;
foreach($subject as $el)
$this->insertBefore($el, $originalFirst);
}
else
$this->insertBefore($subject, $this->firstChild);
return $this;
}
public function append($subject)
{
if(is_array($subject))
{
foreach($subject as $el)
$this->appendChild($subject);
}
else
$this->appendChild($subject);
return $this;
}
public function closest($selector)
{
if($this === $this->ownerDocument->documentElement)
throw new \Exception('Method not valid on document element');
for($el = $this; $el->parentNode != null; $el = $el->parentNode)
{
$m = $el->parentNode->querySelectorAll($selector);
if(array_search($el, $m, true) !== false)
return $el;
}
return null;
}
public function isBefore($other)
{
if($this->parentNode === $other->parentNode)
return ($this->getBreadth() < $other->getBreadth());
$this_depth = $this->getDepth();
$other_depth = $other->getDepth();
if($this_depth == $other_depth)
return $this->parentNode->isBefore($other->parentNode);
if($this_depth > $other_depth)
{
$ancestor = $this;
$ancestor_depth = $this_depth;
while($ancestor_depth > $other_depth)
{
$ancestor = $ancestor->parentNode;
$ancestor_depth--;
}
return $ancestor->isBefore($other);
}
if($this_depth < $other_depth)
{
$ancestor = $other;
$ancestor_depth = $other_depth;
while($ancestor_depth > $this_depth)
{
$ancestor = $ancestor->parentNode;
$ancestor_depth--;
}
return $this->isBefore($ancestor);
}
}
public function getBreadth()
{
$breadth = 0;
for($node = $this->previousSibling; $node != null; $node = $node->previousSibling)
$breadth++;
return $breadth;
}
public function getDepth()
{
$depth = 0;
for($node = $this->parentNode; $node != null; $node = $node->parentNode)
$depth++;
return $depth;
}
private static function sortByDOMPosition($a, $b)
{
return ($a->isBefore($b) ? -1 : 1);
}
public static function selectorToXPath($selector)
{
if(!DOMElement::$xpathConverter)
DOMElement::$xpathConverter = new Selector\XPathConverter();
$xpath = DOMElement::$xpathConverter->convert($selector);
return $xpath;
}
public function import($subject, $forcePHP=false)
{
global $wpgmza;
$node = null;
if($subject instanceof \DOMDocument)
{
if(!$subject->documentElement)
throw new \Exception('Document is empty');
$node = $this->ownerDocument->importNode($subject->documentElement, true);
}
else if($subject instanceof \DOMNode)
{
$node = $this->ownerDocument->importNode($subject, true);
}
else if(preg_match('/(\.html|\.php)$/i', $subject, $m))
{
if(!file_exists($subject))
throw new \Exception('HTML file not found');
$temp = new DOMDocument('1.0', 'UTF-8');
if($forcePHP || preg_match('/\.php$/i', $m[1]))
$temp->loadPHPFile($subject);
else
$temp->load($subject);
$node = $this->ownerDocument->importNode($temp->documentElement, true);
}
else if(is_string($subject))
{
if(empty($subject))
return;
if($subject != strip_tags($subject))
{
$html = DOMDocument::convertUTF8ToHTMLEntities($subject);
$temp = new DOMDocument('1.0', 'UTF-8');
$str = "<div id='domdocument-import-payload___'>" . $subject . "</div>";
if(!empty($wpgmza->settings->developer_mode))
$temp->loadHTML($str);
else
@$temp->loadHTML($str);
$body = $temp->querySelector('#domdocument-import-payload___');
for($child = $body->firstChild; $child != null; $child = $child->nextSibling)
{
$node = $this->ownerDocument->importNode($child, true);
$this->appendChild($node);
}
}
else
$this->appendText($subject);
return;
}
else if(empty($subject))
{
return;
}
else
throw new \Exception('Don\'t know how to import "' . print_r($subject, true) . '" in ' . $this->ownerDocument->documentURI . ' on line ' . $this->getLineNo());
if($body = $node->querySelector("body"))
{
foreach($node->querySelectorAll("body>*") as $child)
$this->appendChild($child);
}
else
$this->appendChild($node);
return $this;
}
public function setInlineStyle($name, $value)
{
$this->removeInlineStyle($name);
$style = $this->getAttribute('style');
$this->setAttribute('style', $style . $name . ':' . $value . ';');
return $this;
}
public function removeInlineStyle($name)
{
if(!$this->hasAttribute('style'))
return;
$style = $this->getAttribute('style');
$rules = preg_split('/\s*;\s*/', $style);
for($i = count($rules) - 1; $i >= 0; $i--)
{
$param = preg_quote($name);
if(preg_match("/^$param\s*:/", trim($rules[$i])))
unset($rules[$i]);
}
$this->setAttribute('style', implode(';', $rules));
return $this;
}
public function hasInlineStyle($name)
{
if(!$this->hasAttribute('style'))
return false;
return preg_match("/\s*$name:.*?((;\s*)|$)/", $this->getAttribute('style'));
}
public function getInlineStyle($name)
{
if(!$this->hasAttribute('style'))
return false;
$m = null;
if(!preg_match("/\s*$name:(.*?)((;\s*)|$)/", $this->getAttribute('style')))
return false;
return $m[1];
}
public function addClass($name)
{
if($this->hasClass($name))
return;
$class = ($this->hasAttribute('class') ? $this->getAttribute('class') : '');
$this->setAttribute('class', $class . (strlen($class) > 0 ? ' ' : '') . $name);
return $this;
}
public function removeClass($name)
{
if(!$this->hasAttribute('class'))
return;
$class = trim(
preg_replace('/\s{2,}/', ' ',
preg_replace('/\\b' . $name . '\\b/', ' ', $this->getAttribute('class'))
)
);
$this->setAttribute('class', $class);
return $this;
}
public function hasClass($name)
{
if(!$this->hasAttribute('class'))
return false;
return preg_match('/\\b' . $name . '\\b/', $this->getAttribute('class'));
}
protected function populateElement($target, $key, $value, $formatters)
{
if(!($target instanceof \DOMElement))
throw new \Exception('Argument must be a DOMElement');
switch(strtolower($target->nodeName))
{
case 'textarea':
case 'select':
case 'input':
$target->setValue($value);
break;
case 'img':
$target->setAttribute('src', $value);
break;
default:
if(!is_null($formatters) && isset($formatters[$key]))
$value = $formatters[$key]($value);
if($value instanceof \DateTime)
$value = $value->format('D jS M \'y g:ia');
if($key == 'price')
$value = number_format($value, 2, '.', '');
if(is_object($value))
throw new \Exception('Expected simple type in "'.$key.'" => "'.print_r($value,true).'"');
$target->import( $value );
break;
}
}
public function populate($src=null, $formatters=null)
{
$x = new \DOMXPath($this->ownerDocument);
if(!$src)
return $this;
if(is_scalar($src))
{
$this->appendText($src);
return $this;
}
foreach($src as $key => $value)
{
if(is_array($value))
{
$m = $x->query('descendant-or-self::*[@name="' . $key . '[]"]', $this);
if($m->length > 0 && count($value) != $m->length)
{
if($src = $m->item(0)->closest('li,tr'))
{
for($i = $m->length; $i < count($value); $i++)
{
$item = $src->cloneNode(true);
$src->parentNode->appendChild($item);
}
$m = $x->query('descendant-or-self::*[@name="' . $key . '[]"]', $this);
}
else
throw new \Exception('Number of elements must match (' . count($value) . ' != ' . $m->length . ')');
}
for($i = 0; $i < $m->length; $i++)
$this->populateElement($m->item($i), $key, $value[$i], $formatters);
}
else
{
$m = $x->query('descendant-or-self::*[@name="' . $key . '" or @data-name="' . $key . '"]', $this);
for($i = 0; $i < $m->length; $i++)
$this->populateElement($m->item($i), $key, $value, $formatters);
}
}
return $this;
}
public function getValue()
{
switch(strtolower($this->nodeName))
{
case 'input':
$type = ($this->hasAttribute('type') ? $this->getAttribute('type') : 'text');
switch($type)
{
case 'radio':
case 'checkbox':
return $this->hasAttribute('checked');
break;
default:
return $this->getAttribute('value');
break;
}
break;
case 'select':
$option = $this->querySelector('option[selected]');
if(!$option)
return null;
if($option->hasAttribute('value'))
return $option->getAttribute('value');
default:
return $this->nodeValue;
break;
}
}
public function setValue($value)
{
switch(strtolower($this->nodeName))
{
case 'textarea':
$this->clear();
$this->appendText( $value );
break;
case 'select':
$deselect = $this->querySelectorAll('option[selected]');
foreach($deselect as $d)
$d->removeAttribute('selected');
if($value === null)
return $this;
$option = $this->querySelector('option[value="' . $value . '"]');
if(!$option)
trigger_error('Option with value "' . $value . '" not found in "' . ($this->getAttribute('name')) . '"', E_USER_WARNING);
else
$option->setAttribute('selected', 'selected');
break;
case 'input':
if(!$this->hasAttribute('type') || $this->getAttribute('type') == 'text')
{
if(is_string($value))
$this->setAttribute('value', $value);
}
else switch(strtolower($this->getAttribute('type')))
{
case 'radio':
if($this->hasAttribute('value') && $this->getAttribute('value') == $value)
$this->setAttribute('checked', 'checked');
else
$this->removeAttribute('checked');
break;
case 'checkbox':
if(!empty($value) && $value != false)
$this->setAttribute('checked', 'checked');
else
$this->removeAttribute('checked');
break;
default:
$this->setAttribute('value', $value);
break;
}
break;
default:
throw new \Exception('Not yet implemented');
$this->nodeValue = $value;
break;
}
return $this;
}
public function appendText($text)
{
$this->appendChild( $this->ownerDocument->createTextNode( $text ) );
return $this;
}
public function insertAfter($elem, $after=null)
{
if($after->parentNode && $after->parentNode !== $this)
throw new \Exception('Hierarchy error');
if($after->nextSibling)
$this->insertBefore($elem, $after->nextSibling);
else
$this->appendChild($elem);
return $this;
}
public function clear()
{
while($this->childNodes->length)
$this->removeChild($this->firstChild);
return $this;
}
public function remove()
{
if($this->parentNode)
$this->parentNode->removeChild($this);
return $this;
}
}
?>