Mercurial > hg > ywww
comparison goodreads/simple_html_dom.php @ 6:077b0a0a3e6d
remaining originals according to dependency walk
| author | Robert Boland <robert@markup.co.uk> |
|---|---|
| date | Thu, 16 Feb 2017 22:29:02 +0000 |
| parents | |
| children |
comparison
equal
deleted
inserted
replaced
| 5:55445b456ad0 | 6:077b0a0a3e6d |
|---|---|
| 1 <?php | |
| 2 /** | |
| 3 * Website: http://sourceforge.net/projects/simplehtmldom/ | |
| 4 * Acknowledge: Jose Solorzano (https://sourceforge.net/projects/php-html/) | |
| 5 * Contributions by: | |
| 6 * Yousuke Kumakura (Attribute filters) | |
| 7 * Vadim Voituk (Negative indexes supports of "find" method) | |
| 8 * Antcs (Constructor with automatically load contents either text or file/url) | |
| 9 * | |
| 10 * all affected sections have comments starting with "PaperG" | |
| 11 * | |
| 12 * Paperg - Added case insensitive testing of the value of the selector. | |
| 13 * Paperg - Added tag_start for the starting index of tags - NOTE: This works but not accurately. | |
| 14 * This tag_start gets counted AFTER \r\n have been crushed out, and after the remove_noice calls so it will not reflect the REAL position of the tag in the source, | |
| 15 * it will almost always be smaller by some amount. | |
| 16 * We use this to determine how far into the file the tag in question is. This "percentage will never be accurate as the $dom->size is the "real" number of bytes the dom was created from. | |
| 17 * but for most purposes, it's a really good estimation. | |
| 18 * Paperg - Added the forceTagsClosed to the dom constructor. Forcing tags closed is great for malformed html, but it CAN lead to parsing errors. | |
| 19 * Allow the user to tell us how much they trust the html. | |
| 20 * Paperg add the text and plaintext to the selectors for the find syntax. plaintext implies text in the innertext of a node. text implies that the tag is a text node. | |
| 21 * This allows for us to find tags based on the text they contain. | |
| 22 * Create find_ancestor_tag to see if a tag is - at any level - inside of another specific tag. | |
| 23 * Paperg: added parse_charset so that we know about the character set of the source document. | |
| 24 * NOTE: If the user's system has a routine called get_last_retrieve_url_contents_content_type availalbe, we will assume it's returning the content-type header from the | |
| 25 * last transfer or curl_exec, and we will parse that and use it in preference to any other method of charset detection. | |
| 26 * | |
| 27 * Licensed under The MIT License | |
| 28 * Redistributions of files must retain the above copyright notice. | |
| 29 * | |
| 30 * @author S.C. Chen <me578022@gmail.com> | |
| 31 * @author John Schlick | |
| 32 * @author Rus Carroll | |
| 33 * @version 1.11 ($Rev: 184 $) | |
| 34 * @package PlaceLocalInclude | |
| 35 * @subpackage simple_html_dom | |
| 36 */ | |
| 37 | |
| 38 /** | |
| 39 * All of the Defines for the classes below. | |
| 40 * @author S.C. Chen <me578022@gmail.com> | |
| 41 */ | |
| 42 define('HDOM_TYPE_ELEMENT', 1); | |
| 43 define('HDOM_TYPE_COMMENT', 2); | |
| 44 define('HDOM_TYPE_TEXT', 3); | |
| 45 define('HDOM_TYPE_ENDTAG', 4); | |
| 46 define('HDOM_TYPE_ROOT', 5); | |
| 47 define('HDOM_TYPE_UNKNOWN', 6); | |
| 48 define('HDOM_QUOTE_DOUBLE', 0); | |
| 49 define('HDOM_QUOTE_SINGLE', 1); | |
| 50 define('HDOM_QUOTE_NO', 3); | |
| 51 define('HDOM_INFO_BEGIN', 0); | |
| 52 define('HDOM_INFO_END', 1); | |
| 53 define('HDOM_INFO_QUOTE', 2); | |
| 54 define('HDOM_INFO_SPACE', 3); | |
| 55 define('HDOM_INFO_TEXT', 4); | |
| 56 define('HDOM_INFO_INNER', 5); | |
| 57 define('HDOM_INFO_OUTER', 6); | |
| 58 define('HDOM_INFO_ENDSPACE',7); | |
| 59 define('DEFAULT_TARGET_CHARSET', 'UTF-8'); | |
| 60 define('DEFAULT_BR_TEXT', "\r\n"); | |
| 61 // helper functions | |
| 62 // ----------------------------------------------------------------------------- | |
| 63 // get html dom from file | |
| 64 // $maxlen is defined in the code as PHP_STREAM_COPY_ALL which is defined as -1. | |
| 65 function file_get_html($url, $use_include_path = false, $context=null, $offset = -1, $maxLen=-1, $lowercase = true, $forceTagsClosed=true, $target_charset = DEFAULT_TARGET_CHARSET, $stripRN=true, $defaultBRText=DEFAULT_BR_TEXT) | |
| 66 { | |
| 67 // We DO force the tags to be terminated. | |
| 68 $dom = new simple_html_dom(null, $lowercase, $forceTagsClosed, $target_charset, $defaultBRText); | |
| 69 // For sourceforge users: uncomment the next line and comment the retreive_url_contents line 2 lines down if it is not already done. | |
| 70 $contents = file_get_contents($url, $use_include_path, $context, $offset); | |
| 71 // Paperg - use our own mechanism for getting the contents as we want to control the timeout. | |
| 72 // $contents = retrieve_url_contents($url); | |
| 73 if (empty($contents)) | |
| 74 { | |
| 75 return false; | |
| 76 } | |
| 77 // The second parameter can force the selectors to all be lowercase. | |
| 78 $dom->load($contents, $lowercase, $stripRN); | |
| 79 return $dom; | |
| 80 } | |
| 81 | |
| 82 // get html dom from string | |
| 83 function str_get_html($str, $lowercase=true, $forceTagsClosed=true, $target_charset = DEFAULT_TARGET_CHARSET, $stripRN=true, $defaultBRText=DEFAULT_BR_TEXT) | |
| 84 { | |
| 85 $dom = new simple_html_dom(null, $lowercase, $forceTagsClosed, $target_charset, $defaultBRText); | |
| 86 if (empty($str)) | |
| 87 { | |
| 88 $dom->clear(); | |
| 89 return false; | |
| 90 } | |
| 91 $dom->load($str, $lowercase, $stripRN); | |
| 92 return $dom; | |
| 93 } | |
| 94 | |
| 95 // dump html dom tree | |
| 96 function dump_html_tree($node, $show_attr=true, $deep=0) | |
| 97 { | |
| 98 $node->dump($node); | |
| 99 } | |
| 100 | |
| 101 /** | |
| 102 * simple html dom node | |
| 103 * PaperG - added ability for "find" routine to lowercase the value of the selector. | |
| 104 * PaperG - added $tag_start to track the start position of the tag in the total byte index | |
| 105 * | |
| 106 * @package PlaceLocalInclude | |
| 107 */ | |
| 108 class simple_html_dom_node { | |
| 109 public $nodetype = HDOM_TYPE_TEXT; | |
| 110 public $tag = 'text'; | |
| 111 public $attr = array(); | |
| 112 public $children = array(); | |
| 113 public $nodes = array(); | |
| 114 public $parent = null; | |
| 115 public $_ = array(); | |
| 116 public $tag_start = 0; | |
| 117 private $dom = null; | |
| 118 | |
| 119 function __construct($dom) | |
| 120 { | |
| 121 $this->dom = $dom; | |
| 122 $dom->nodes[] = $this; | |
| 123 } | |
| 124 | |
| 125 function __destruct() | |
| 126 { | |
| 127 $this->clear(); | |
| 128 } | |
| 129 | |
| 130 function __toString() | |
| 131 { | |
| 132 return $this->outertext(); | |
| 133 } | |
| 134 | |
| 135 // clean up memory due to php5 circular references memory leak... | |
| 136 function clear() | |
| 137 { | |
| 138 $this->dom = null; | |
| 139 $this->nodes = null; | |
| 140 $this->parent = null; | |
| 141 $this->children = null; | |
| 142 } | |
| 143 | |
| 144 // dump node's tree | |
| 145 function dump($show_attr=true, $deep=0) | |
| 146 { | |
| 147 $lead = str_repeat(' ', $deep); | |
| 148 | |
| 149 echo $lead.$this->tag; | |
| 150 if ($show_attr && count($this->attr)>0) | |
| 151 { | |
| 152 echo '('; | |
| 153 foreach ($this->attr as $k=>$v) | |
| 154 echo "[$k]=>\"".$this->$k.'", '; | |
| 155 echo ')'; | |
| 156 } | |
| 157 echo "\n"; | |
| 158 | |
| 159 foreach ($this->nodes as $c) | |
| 160 $c->dump($show_attr, $deep+1); | |
| 161 } | |
| 162 | |
| 163 | |
| 164 // Debugging function to dump a single dom node with a bunch of information about it. | |
| 165 function dump_node() | |
| 166 { | |
| 167 echo $this->tag; | |
| 168 if (count($this->attr)>0) | |
| 169 { | |
| 170 echo '('; | |
| 171 foreach ($this->attr as $k=>$v) | |
| 172 { | |
| 173 echo "[$k]=>\"".$this->$k.'", '; | |
| 174 } | |
| 175 echo ')'; | |
| 176 } | |
| 177 if (count($this->attr)>0) | |
| 178 { | |
| 179 echo ' $_ ('; | |
| 180 foreach ($this->_ as $k=>$v) | |
| 181 { | |
| 182 if (is_array($v)) | |
| 183 { | |
| 184 echo "[$k]=>("; | |
| 185 foreach ($v as $k2=>$v2) | |
| 186 { | |
| 187 echo "[$k2]=>\"".$v2.'", '; | |
| 188 } | |
| 189 echo ")"; | |
| 190 } else { | |
| 191 echo "[$k]=>\"".$v.'", '; | |
| 192 } | |
| 193 } | |
| 194 echo ")"; | |
| 195 } | |
| 196 | |
| 197 if (isset($this->text)) | |
| 198 { | |
| 199 echo " text: (" . $this->text . ")"; | |
| 200 } | |
| 201 | |
| 202 echo " children: " . count($this->children); | |
| 203 echo " nodes: " . count($this->nodes); | |
| 204 echo " tag_start: " . $this->tag_start; | |
| 205 echo "\n"; | |
| 206 | |
| 207 } | |
| 208 | |
| 209 // returns the parent of node | |
| 210 function parent() | |
| 211 { | |
| 212 return $this->parent; | |
| 213 } | |
| 214 | |
| 215 // returns children of node | |
| 216 function children($idx=-1) | |
| 217 { | |
| 218 if ($idx===-1) return $this->children; | |
| 219 if (isset($this->children[$idx])) return $this->children[$idx]; | |
| 220 return null; | |
| 221 } | |
| 222 | |
| 223 // returns the first child of node | |
| 224 function first_child() | |
| 225 { | |
| 226 if (count($this->children)>0) return $this->children[0]; | |
| 227 return null; | |
| 228 } | |
| 229 | |
| 230 // returns the last child of node | |
| 231 function last_child() | |
| 232 { | |
| 233 if (($count=count($this->children))>0) return $this->children[$count-1]; | |
| 234 return null; | |
| 235 } | |
| 236 | |
| 237 // returns the next sibling of node | |
| 238 function next_sibling() | |
| 239 { | |
| 240 if ($this->parent===null) return null; | |
| 241 $idx = 0; | |
| 242 $count = count($this->parent->children); | |
| 243 while ($idx<$count && $this!==$this->parent->children[$idx]) | |
| 244 ++$idx; | |
| 245 if (++$idx>=$count) return null; | |
| 246 return $this->parent->children[$idx]; | |
| 247 } | |
| 248 | |
| 249 // returns the previous sibling of node | |
| 250 function prev_sibling() | |
| 251 { | |
| 252 if ($this->parent===null) return null; | |
| 253 $idx = 0; | |
| 254 $count = count($this->parent->children); | |
| 255 while ($idx<$count && $this!==$this->parent->children[$idx]) | |
| 256 ++$idx; | |
| 257 if (--$idx<0) return null; | |
| 258 return $this->parent->children[$idx]; | |
| 259 } | |
| 260 | |
| 261 // function to locate a specific ancestor tag in the path to the root. | |
| 262 function find_ancestor_tag($tag) | |
| 263 { | |
| 264 global $debugObject; | |
| 265 if (is_object($debugObject)) | |
| 266 { | |
| 267 $debugObject->debugLogEntry(1); | |
| 268 } | |
| 269 | |
| 270 // Start by including ourselves in the comparison. | |
| 271 $returnDom = $this; | |
| 272 | |
| 273 while (!is_null($returnDom)) | |
| 274 { | |
| 275 if (is_object($debugObject)) | |
| 276 { | |
| 277 $debugObject->debugLog(2, "Current tag is: " . $returnDom->tag); | |
| 278 } | |
| 279 | |
| 280 if ($returnDom->tag == $tag) | |
| 281 { | |
| 282 break; | |
| 283 } | |
| 284 $returnDom = $returnDom->parent; | |
| 285 } | |
| 286 return $returnDom; | |
| 287 } | |
| 288 | |
| 289 // get dom node's inner html | |
| 290 function innertext() | |
| 291 { | |
| 292 if (isset($this->_[HDOM_INFO_INNER])) return $this->_[HDOM_INFO_INNER]; | |
| 293 if (isset($this->_[HDOM_INFO_TEXT])) return $this->dom->restore_noise($this->_[HDOM_INFO_TEXT]); | |
| 294 | |
| 295 $ret = ''; | |
| 296 foreach ($this->nodes as $n) | |
| 297 $ret .= $n->outertext(); | |
| 298 return $ret; | |
| 299 } | |
| 300 | |
| 301 // get dom node's outer text (with tag) | |
| 302 function outertext() | |
| 303 { | |
| 304 global $debugObject; | |
| 305 if (is_object($debugObject)) | |
| 306 { | |
| 307 $text = ''; | |
| 308 if ($this->tag == 'text') | |
| 309 { | |
| 310 if (!empty($this->text)) | |
| 311 { | |
| 312 $text = " with text: " . $this->text; | |
| 313 } | |
| 314 } | |
| 315 $debugObject->debugLog(1, 'Innertext of tag: ' . $this->tag . $text); | |
| 316 } | |
| 317 | |
| 318 if ($this->tag==='root') return $this->innertext(); | |
| 319 | |
| 320 // trigger callback | |
| 321 if ($this->dom && $this->dom->callback!==null) | |
| 322 { | |
| 323 call_user_func_array($this->dom->callback, array($this)); | |
| 324 } | |
| 325 | |
| 326 if (isset($this->_[HDOM_INFO_OUTER])) return $this->_[HDOM_INFO_OUTER]; | |
| 327 if (isset($this->_[HDOM_INFO_TEXT])) return $this->dom->restore_noise($this->_[HDOM_INFO_TEXT]); | |
| 328 | |
| 329 // render begin tag | |
| 330 if ($this->dom && $this->dom->nodes[$this->_[HDOM_INFO_BEGIN]]) | |
| 331 { | |
| 332 $ret = $this->dom->nodes[$this->_[HDOM_INFO_BEGIN]]->makeup(); | |
| 333 } else { | |
| 334 $ret = ""; | |
| 335 } | |
| 336 | |
| 337 // render inner text | |
| 338 if (isset($this->_[HDOM_INFO_INNER])) | |
| 339 { | |
| 340 // If it's a br tag... don't return the HDOM_INNER_INFO that we may or may not have added. | |
| 341 if ($this->tag != "br") | |
| 342 { | |
| 343 $ret .= $this->_[HDOM_INFO_INNER]; | |
| 344 } | |
| 345 } else { | |
| 346 if ($this->nodes) | |
| 347 { | |
| 348 foreach ($this->nodes as $n) | |
| 349 { | |
| 350 $ret .= $this->convert_text($n->outertext()); | |
| 351 } | |
| 352 } | |
| 353 } | |
| 354 | |
| 355 // render end tag | |
| 356 if (isset($this->_[HDOM_INFO_END]) && $this->_[HDOM_INFO_END]!=0) | |
| 357 $ret .= '</'.$this->tag.'>'; | |
| 358 return $ret; | |
| 359 } | |
| 360 | |
| 361 // get dom node's plain text | |
| 362 function text() | |
| 363 { | |
| 364 if (isset($this->_[HDOM_INFO_INNER])) return $this->_[HDOM_INFO_INNER]; | |
| 365 switch ($this->nodetype) | |
| 366 { | |
| 367 case HDOM_TYPE_TEXT: return $this->dom->restore_noise($this->_[HDOM_INFO_TEXT]); | |
| 368 case HDOM_TYPE_COMMENT: return ''; | |
| 369 case HDOM_TYPE_UNKNOWN: return ''; | |
| 370 } | |
| 371 if (strcasecmp($this->tag, 'script')===0) return ''; | |
| 372 if (strcasecmp($this->tag, 'style')===0) return ''; | |
| 373 | |
| 374 $ret = ''; | |
| 375 // In rare cases, (always node type 1 or HDOM_TYPE_ELEMENT - observed for some span tags, and some p tags) $this->nodes is set to NULL. | |
| 376 // NOTE: This indicates that there is a problem where it's set to NULL without a clear happening. | |
| 377 // WHY is this happening? | |
| 378 if (!is_null($this->nodes)) | |
| 379 { | |
| 380 foreach ($this->nodes as $n) | |
| 381 { | |
| 382 $ret .= $this->convert_text($n->text()); | |
| 383 } | |
| 384 } | |
| 385 return $ret; | |
| 386 } | |
| 387 | |
| 388 function xmltext() | |
| 389 { | |
| 390 $ret = $this->innertext(); | |
| 391 $ret = str_ireplace('<![CDATA[', '', $ret); | |
| 392 $ret = str_replace(']]>', '', $ret); | |
| 393 return $ret; | |
| 394 } | |
| 395 | |
| 396 // build node's text with tag | |
| 397 function makeup() | |
| 398 { | |
| 399 // text, comment, unknown | |
| 400 if (isset($this->_[HDOM_INFO_TEXT])) return $this->dom->restore_noise($this->_[HDOM_INFO_TEXT]); | |
| 401 | |
| 402 $ret = '<'.$this->tag; | |
| 403 $i = -1; | |
| 404 | |
| 405 foreach ($this->attr as $key=>$val) | |
| 406 { | |
| 407 ++$i; | |
| 408 | |
| 409 // skip removed attribute | |
| 410 if ($val===null || $val===false) | |
| 411 continue; | |
| 412 | |
| 413 $ret .= $this->_[HDOM_INFO_SPACE][$i][0]; | |
| 414 //no value attr: nowrap, checked selected... | |
| 415 if ($val===true) | |
| 416 $ret .= $key; | |
| 417 else { | |
| 418 switch ($this->_[HDOM_INFO_QUOTE][$i]) | |
| 419 { | |
| 420 case HDOM_QUOTE_DOUBLE: $quote = '"'; break; | |
| 421 case HDOM_QUOTE_SINGLE: $quote = '\''; break; | |
| 422 default: $quote = ''; | |
| 423 } | |
| 424 $ret .= $key.$this->_[HDOM_INFO_SPACE][$i][1].'='.$this->_[HDOM_INFO_SPACE][$i][2].$quote.$val.$quote; | |
| 425 } | |
| 426 } | |
| 427 $ret = $this->dom->restore_noise($ret); | |
| 428 return $ret . $this->_[HDOM_INFO_ENDSPACE] . '>'; | |
| 429 } | |
| 430 | |
| 431 // find elements by css selector | |
| 432 //PaperG - added ability for find to lowercase the value of the selector. | |
| 433 function find($selector, $idx=null, $lowercase=false) | |
| 434 { | |
| 435 $selectors = $this->parse_selector($selector); | |
| 436 if (($count=count($selectors))===0) return array(); | |
| 437 $found_keys = array(); | |
| 438 | |
| 439 // find each selector | |
| 440 for ($c=0; $c<$count; ++$c) | |
| 441 { | |
| 442 // The change on the below line was documented on the sourceforge code tracker id 2788009 | |
| 443 // used to be: if (($levle=count($selectors[0]))===0) return array(); | |
| 444 if (($levle=count($selectors[$c]))===0) return array(); | |
| 445 if (!isset($this->_[HDOM_INFO_BEGIN])) return array(); | |
| 446 | |
| 447 $head = array($this->_[HDOM_INFO_BEGIN]=>1); | |
| 448 | |
| 449 // handle descendant selectors, no recursive! | |
| 450 for ($l=0; $l<$levle; ++$l) | |
| 451 { | |
| 452 $ret = array(); | |
| 453 foreach ($head as $k=>$v) | |
| 454 { | |
| 455 $n = ($k===-1) ? $this->dom->root : $this->dom->nodes[$k]; | |
| 456 //PaperG - Pass this optional parameter on to the seek function. | |
| 457 $n->seek($selectors[$c][$l], $ret, $lowercase); | |
| 458 } | |
| 459 $head = $ret; | |
| 460 } | |
| 461 | |
| 462 foreach ($head as $k=>$v) | |
| 463 { | |
| 464 if (!isset($found_keys[$k])) | |
| 465 $found_keys[$k] = 1; | |
| 466 } | |
| 467 } | |
| 468 | |
| 469 // sort keys | |
| 470 ksort($found_keys); | |
| 471 | |
| 472 $found = array(); | |
| 473 foreach ($found_keys as $k=>$v) | |
| 474 $found[] = $this->dom->nodes[$k]; | |
| 475 | |
| 476 // return nth-element or array | |
| 477 if (is_null($idx)) return $found; | |
| 478 else if ($idx<0) $idx = count($found) + $idx; | |
| 479 return (isset($found[$idx])) ? $found[$idx] : null; | |
| 480 } | |
| 481 | |
| 482 // seek for given conditions | |
| 483 // PaperG - added parameter to allow for case insensitive testing of the value of a selector. | |
| 484 protected function seek($selector, &$ret, $lowercase=false) | |
| 485 { | |
| 486 global $debugObject; | |
| 487 if (is_object($debugObject)) | |
| 488 { | |
| 489 $debugObject->debugLogEntry(1); | |
| 490 } | |
| 491 | |
| 492 list($tag, $key, $val, $exp, $no_key) = $selector; | |
| 493 | |
| 494 // xpath index | |
| 495 if ($tag && $key && is_numeric($key)) | |
| 496 { | |
| 497 $count = 0; | |
| 498 foreach ($this->children as $c) | |
| 499 { | |
| 500 if ($tag==='*' || $tag===$c->tag) { | |
| 501 if (++$count==$key) { | |
| 502 $ret[$c->_[HDOM_INFO_BEGIN]] = 1; | |
| 503 return; | |
| 504 } | |
| 505 } | |
| 506 } | |
| 507 return; | |
| 508 } | |
| 509 | |
| 510 $end = (!empty($this->_[HDOM_INFO_END])) ? $this->_[HDOM_INFO_END] : 0; | |
| 511 if ($end==0) { | |
| 512 $parent = $this->parent; | |
| 513 while (!isset($parent->_[HDOM_INFO_END]) && $parent!==null) { | |
| 514 $end -= 1; | |
| 515 $parent = $parent->parent; | |
| 516 } | |
| 517 $end += $parent->_[HDOM_INFO_END]; | |
| 518 } | |
| 519 | |
| 520 for ($i=$this->_[HDOM_INFO_BEGIN]+1; $i<$end; ++$i) { | |
| 521 $node = $this->dom->nodes[$i]; | |
| 522 | |
| 523 $pass = true; | |
| 524 | |
| 525 if ($tag==='*' && !$key) { | |
| 526 if (in_array($node, $this->children, true)) | |
| 527 $ret[$i] = 1; | |
| 528 continue; | |
| 529 } | |
| 530 | |
| 531 // compare tag | |
| 532 if ($tag && $tag!=$node->tag && $tag!=='*') {$pass=false;} | |
| 533 // compare key | |
| 534 if ($pass && $key) { | |
| 535 if ($no_key) { | |
| 536 if (isset($node->attr[$key])) $pass=false; | |
| 537 } else { | |
| 538 if (($key != "plaintext") && !isset($node->attr[$key])) $pass=false; | |
| 539 } | |
| 540 } | |
| 541 // compare value | |
| 542 if ($pass && $key && $val && $val!=='*') { | |
| 543 // If they have told us that this is a "plaintext" search then we want the plaintext of the node - right? | |
| 544 if ($key == "plaintext") { | |
| 545 // $node->plaintext actually returns $node->text(); | |
| 546 $nodeKeyValue = $node->text(); | |
| 547 } else { | |
| 548 // this is a normal search, we want the value of that attribute of the tag. | |
| 549 $nodeKeyValue = $node->attr[$key]; | |
| 550 } | |
| 551 if (is_object($debugObject)) {$debugObject->debugLog(2, "testing node: " . $node->tag . " for attribute: " . $key . $exp . $val . " where nodes value is: " . $nodeKeyValue);} | |
| 552 | |
| 553 //PaperG - If lowercase is set, do a case insensitive test of the value of the selector. | |
| 554 if ($lowercase) { | |
| 555 $check = $this->match($exp, strtolower($val), strtolower($nodeKeyValue)); | |
| 556 } else { | |
| 557 $check = $this->match($exp, $val, $nodeKeyValue); | |
| 558 } | |
| 559 if (is_object($debugObject)) {$debugObject->debugLog(2, "after match: " . ($check ? "true" : "false"));} | |
| 560 | |
| 561 // handle multiple class | |
| 562 if (!$check && strcasecmp($key, 'class')===0) { | |
| 563 foreach (explode(' ',$node->attr[$key]) as $k) { | |
| 564 // Without this, there were cases where leading, trailing, or double spaces lead to our comparing blanks - bad form. | |
| 565 if (!empty($k)) { | |
| 566 if ($lowercase) { | |
| 567 $check = $this->match($exp, strtolower($val), strtolower($k)); | |
| 568 } else { | |
| 569 $check = $this->match($exp, $val, $k); | |
| 570 } | |
| 571 if ($check) break; | |
| 572 } | |
| 573 } | |
| 574 } | |
| 575 if (!$check) $pass = false; | |
| 576 } | |
| 577 if ($pass) $ret[$i] = 1; | |
| 578 unset($node); | |
| 579 } | |
| 580 // It's passed by reference so this is actually what this function returns. | |
| 581 if (is_object($debugObject)) {$debugObject->debugLog(1, "EXIT - ret: ", $ret);} | |
| 582 } | |
| 583 | |
| 584 protected function match($exp, $pattern, $value) { | |
| 585 global $debugObject; | |
| 586 if (is_object($debugObject)) {$debugObject->debugLogEntry(1);} | |
| 587 | |
| 588 switch ($exp) { | |
| 589 case '=': | |
| 590 return ($value===$pattern); | |
| 591 case '!=': | |
| 592 return ($value!==$pattern); | |
| 593 case '^=': | |
| 594 return preg_match("/^".preg_quote($pattern,'/')."/", $value); | |
| 595 case '$=': | |
| 596 return preg_match("/".preg_quote($pattern,'/')."$/", $value); | |
| 597 case '*=': | |
| 598 if ($pattern[0]=='/') { | |
| 599 return preg_match($pattern, $value); | |
| 600 } | |
| 601 return preg_match("/".$pattern."/i", $value); | |
| 602 } | |
| 603 return false; | |
| 604 } | |
| 605 | |
| 606 protected function parse_selector($selector_string) { | |
| 607 global $debugObject; | |
| 608 if (is_object($debugObject)) {$debugObject->debugLogEntry(1);} | |
| 609 | |
| 610 // pattern of CSS selectors, modified from mootools | |
| 611 // Paperg: Add the colon to the attrbute, so that it properly finds <tag attr:ibute="something" > like google does. | |
| 612 // Note: if you try to look at this attribute, yo MUST use getAttribute since $dom->x:y will fail the php syntax check. | |
| 613 // Notice the \[ starting the attbute? and the @? following? This implies that an attribute can begin with an @ sign that is not captured. | |
| 614 // This implies that an html attribute specifier may start with an @ sign that is NOT captured by the expression. | |
| 615 // farther study is required to determine of this should be documented or removed. | |
| 616 // $pattern = "/([\w-:\*]*)(?:\#([\w-]+)|\.([\w-]+))?(?:\[@?(!?[\w-]+)(?:([!*^$]?=)[\"']?(.*?)[\"']?)?\])?([\/, ]+)/is"; | |
| 617 $pattern = "/([\w-:\*]*)(?:\#([\w-]+)|\.([\w-]+))?(?:\[@?(!?[\w-:]+)(?:([!*^$]?=)[\"']?(.*?)[\"']?)?\])?([\/, ]+)/is"; | |
| 618 preg_match_all($pattern, trim($selector_string).' ', $matches, PREG_SET_ORDER); | |
| 619 if (is_object($debugObject)) {$debugObject->debugLog(2, "Matches Array: ", $matches);} | |
| 620 | |
| 621 $selectors = array(); | |
| 622 $result = array(); | |
| 623 //print_r($matches); | |
| 624 | |
| 625 foreach ($matches as $m) { | |
| 626 $m[0] = trim($m[0]); | |
| 627 if ($m[0]==='' || $m[0]==='/' || $m[0]==='//') continue; | |
| 628 // for browser generated xpath | |
| 629 if ($m[1]==='tbody') continue; | |
| 630 | |
| 631 list($tag, $key, $val, $exp, $no_key) = array($m[1], null, null, '=', false); | |
| 632 if (!empty($m[2])) {$key='id'; $val=$m[2];} | |
| 633 if (!empty($m[3])) {$key='class'; $val=$m[3];} | |
| 634 if (!empty($m[4])) {$key=$m[4];} | |
| 635 if (!empty($m[5])) {$exp=$m[5];} | |
| 636 if (!empty($m[6])) {$val=$m[6];} | |
| 637 | |
| 638 // convert to lowercase | |
| 639 if ($this->dom->lowercase) {$tag=strtolower($tag); $key=strtolower($key);} | |
| 640 //elements that do NOT have the specified attribute | |
| 641 if (isset($key[0]) && $key[0]==='!') {$key=substr($key, 1); $no_key=true;} | |
| 642 | |
| 643 $result[] = array($tag, $key, $val, $exp, $no_key); | |
| 644 if (trim($m[7])===',') { | |
| 645 $selectors[] = $result; | |
| 646 $result = array(); | |
| 647 } | |
| 648 } | |
| 649 if (count($result)>0) | |
| 650 $selectors[] = $result; | |
| 651 return $selectors; | |
| 652 } | |
| 653 | |
| 654 function __get($name) { | |
| 655 if (isset($this->attr[$name])) | |
| 656 { | |
| 657 return $this->convert_text($this->attr[$name]); | |
| 658 } | |
| 659 switch ($name) { | |
| 660 case 'outertext': return $this->outertext(); | |
| 661 case 'innertext': return $this->innertext(); | |
| 662 case 'plaintext': return $this->text(); | |
| 663 case 'xmltext': return $this->xmltext(); | |
| 664 default: return array_key_exists($name, $this->attr); | |
| 665 } | |
| 666 } | |
| 667 | |
| 668 function __set($name, $value) { | |
| 669 switch ($name) { | |
| 670 case 'outertext': return $this->_[HDOM_INFO_OUTER] = $value; | |
| 671 case 'innertext': | |
| 672 if (isset($this->_[HDOM_INFO_TEXT])) return $this->_[HDOM_INFO_TEXT] = $value; | |
| 673 return $this->_[HDOM_INFO_INNER] = $value; | |
| 674 } | |
| 675 if (!isset($this->attr[$name])) { | |
| 676 $this->_[HDOM_INFO_SPACE][] = array(' ', '', ''); | |
| 677 $this->_[HDOM_INFO_QUOTE][] = HDOM_QUOTE_DOUBLE; | |
| 678 } | |
| 679 $this->attr[$name] = $value; | |
| 680 } | |
| 681 | |
| 682 function __isset($name) { | |
| 683 switch ($name) { | |
| 684 case 'outertext': return true; | |
| 685 case 'innertext': return true; | |
| 686 case 'plaintext': return true; | |
| 687 } | |
| 688 //no value attr: nowrap, checked selected... | |
| 689 return (array_key_exists($name, $this->attr)) ? true : isset($this->attr[$name]); | |
| 690 } | |
| 691 | |
| 692 function __unset($name) { | |
| 693 if (isset($this->attr[$name])) | |
| 694 unset($this->attr[$name]); | |
| 695 } | |
| 696 | |
| 697 // PaperG - Function to convert the text from one character set to another if the two sets are not the same. | |
| 698 function convert_text($text) { | |
| 699 global $debugObject; | |
| 700 if (is_object($debugObject)) {$debugObject->debugLogEntry(1);} | |
| 701 | |
| 702 $converted_text = $text; | |
| 703 | |
| 704 $sourceCharset = ""; | |
| 705 $targetCharset = ""; | |
| 706 if ($this->dom) { | |
| 707 $sourceCharset = strtoupper($this->dom->_charset); | |
| 708 $targetCharset = strtoupper($this->dom->_target_charset); | |
| 709 } | |
| 710 if (is_object($debugObject)) {$debugObject->debugLog(3, "source charset: " . $sourceCharset . " target charaset: " . $targetCharset);} | |
| 711 | |
| 712 if (!empty($sourceCharset) && !empty($targetCharset) && (strcasecmp($sourceCharset, $targetCharset) != 0)) | |
| 713 { | |
| 714 // Check if the reported encoding could have been incorrect and the text is actually already UTF-8 | |
| 715 if ((strcasecmp($targetCharset, 'UTF-8') == 0) && ($this->is_utf8($text))) | |
| 716 { | |
| 717 $converted_text = $text; | |
| 718 } | |
| 719 else | |
| 720 { | |
| 721 $converted_text = iconv($sourceCharset, $targetCharset, $text); | |
| 722 } | |
| 723 } | |
| 724 | |
| 725 return $converted_text; | |
| 726 } | |
| 727 | |
| 728 function is_utf8($string) | |
| 729 { | |
| 730 return (utf8_encode(utf8_decode($string)) == $string); | |
| 731 } | |
| 732 | |
| 733 // camel naming conventions | |
| 734 function getAllAttributes() {return $this->attr;} | |
| 735 function getAttribute($name) {return $this->__get($name);} | |
| 736 function setAttribute($name, $value) {$this->__set($name, $value);} | |
| 737 function hasAttribute($name) {return $this->__isset($name);} | |
| 738 function removeAttribute($name) {$this->__set($name, null);} | |
| 739 function getElementById($id) {return $this->find("#$id", 0);} | |
| 740 function getElementsById($id, $idx=null) {return $this->find("#$id", $idx);} | |
| 741 function getElementByTagName($name) {return $this->find($name, 0);} | |
| 742 function getElementsByTagName($name, $idx=null) {return $this->find($name, $idx);} | |
| 743 function parentNode() {return $this->parent();} | |
| 744 function childNodes($idx=-1) {return $this->children($idx);} | |
| 745 function firstChild() {return $this->first_child();} | |
| 746 function lastChild() {return $this->last_child();} | |
| 747 function nextSibling() {return $this->next_sibling();} | |
| 748 function previousSibling() {return $this->prev_sibling();} | |
| 749 } | |
| 750 | |
| 751 /** | |
| 752 * simple html dom parser | |
| 753 * Paperg - in the find routine: allow us to specify that we want case insensitive testing of the value of the selector. | |
| 754 * Paperg - change $size from protected to public so we can easily access it | |
| 755 * Paperg - added ForceTagsClosed in the constructor which tells us whether we trust the html or not. Default is to NOT trust it. | |
| 756 * | |
| 757 * @package PlaceLocalInclude | |
| 758 */ | |
| 759 class simple_html_dom { | |
| 760 public $root = null; | |
| 761 public $nodes = array(); | |
| 762 public $callback = null; | |
| 763 public $lowercase = false; | |
| 764 public $size; | |
| 765 protected $pos; | |
| 766 protected $doc; | |
| 767 protected $char; | |
| 768 protected $cursor; | |
| 769 protected $parent; | |
| 770 protected $noise = array(); | |
| 771 protected $token_blank = " \t\r\n"; | |
| 772 protected $token_equal = ' =/>'; | |
| 773 protected $token_slash = " />\r\n\t"; | |
| 774 protected $token_attr = ' >'; | |
| 775 protected $_charset = ''; | |
| 776 protected $_target_charset = ''; | |
| 777 protected $default_br_text = ""; | |
| 778 | |
| 779 // use isset instead of in_array, performance boost about 30%... | |
| 780 protected $self_closing_tags = array('img'=>1, 'br'=>1, 'input'=>1, 'meta'=>1, 'link'=>1, 'hr'=>1, 'base'=>1, 'embed'=>1, 'spacer'=>1); | |
| 781 protected $block_tags = array('root'=>1, 'body'=>1, 'form'=>1, 'div'=>1, 'span'=>1, 'table'=>1); | |
| 782 // Known sourceforge issue #2977341 | |
| 783 // B tags that are not closed cause us to return everything to the end of the document. | |
| 784 protected $optional_closing_tags = array( | |
| 785 'tr'=>array('tr'=>1, 'td'=>1, 'th'=>1), | |
| 786 'th'=>array('th'=>1), | |
| 787 'td'=>array('td'=>1), | |
| 788 'li'=>array('li'=>1), | |
| 789 'dt'=>array('dt'=>1, 'dd'=>1), | |
| 790 'dd'=>array('dd'=>1, 'dt'=>1), | |
| 791 'dl'=>array('dd'=>1, 'dt'=>1), | |
| 792 'p'=>array('p'=>1), | |
| 793 'nobr'=>array('nobr'=>1), | |
| 794 'b'=>array('b'=>1), | |
| 795 ); | |
| 796 | |
| 797 function __construct($str=null, $lowercase=true, $forceTagsClosed=true, $target_charset=DEFAULT_TARGET_CHARSET, $stripRN=true, $defaultBRText=DEFAULT_BR_TEXT) { | |
| 798 if ($str) { | |
| 799 if (preg_match("/^http:\/\//i",$str) || is_file($str)) | |
| 800 $this->load_file($str); | |
| 801 else | |
| 802 $this->load($str, $lowercase, $stripRN, $defaultBRText); | |
| 803 } | |
| 804 // Forcing tags to be closed implies that we don't trust the html, but it can lead to parsing errors if we SHOULD trust the html. | |
| 805 if (!$forceTagsClosed) { | |
| 806 $this->optional_closing_array=array(); | |
| 807 } | |
| 808 $this->_target_charset = $target_charset; | |
| 809 } | |
| 810 | |
| 811 function __destruct() { | |
| 812 $this->clear(); | |
| 813 } | |
| 814 | |
| 815 // load html from string | |
| 816 function load($str, $lowercase=true, $stripRN=true, $defaultBRText=DEFAULT_BR_TEXT) { | |
| 817 global $debugObject; | |
| 818 | |
| 819 // prepare | |
| 820 $this->prepare($str, $lowercase, $stripRN, $defaultBRText); | |
| 821 // strip out comments | |
| 822 $this->remove_noise("'<!--(.*?)-->'is"); | |
| 823 // strip out cdata | |
| 824 $this->remove_noise("'<!\[CDATA\[(.*?)\]\]>'is", true); | |
| 825 // Per sourceforge http://sourceforge.net/tracker/?func=detail&aid=2949097&group_id=218559&atid=1044037 | |
| 826 // Script tags removal now preceeds style tag removal. | |
| 827 // strip out <script> tags | |
| 828 $this->remove_noise("'<\s*script[^>]*[^/]>(.*?)<\s*/\s*script\s*>'is"); | |
| 829 $this->remove_noise("'<\s*script\s*>(.*?)<\s*/\s*script\s*>'is"); | |
| 830 // strip out <style> tags | |
| 831 $this->remove_noise("'<\s*style[^>]*[^/]>(.*?)<\s*/\s*style\s*>'is"); | |
| 832 $this->remove_noise("'<\s*style\s*>(.*?)<\s*/\s*style\s*>'is"); | |
| 833 // strip out preformatted tags | |
| 834 $this->remove_noise("'<\s*(?:code)[^>]*>(.*?)<\s*/\s*(?:code)\s*>'is"); | |
| 835 // strip out server side scripts | |
| 836 $this->remove_noise("'(<\?)(.*?)(\?>)'s", true); | |
| 837 // strip smarty scripts | |
| 838 $this->remove_noise("'(\{\w)(.*?)(\})'s", true); | |
| 839 | |
| 840 // parsing | |
| 841 while ($this->parse()); | |
| 842 // end | |
| 843 $this->root->_[HDOM_INFO_END] = $this->cursor; | |
| 844 $this->parse_charset(); | |
| 845 } | |
| 846 | |
| 847 // load html from file | |
| 848 function load_file() { | |
| 849 $args = func_get_args(); | |
| 850 $this->load(call_user_func_array('file_get_contents', $args), true); | |
| 851 // Per the simple_html_dom repositiry this is a planned upgrade to the codebase. | |
| 852 // Throw an error if we can't properly load the dom. | |
| 853 if (($error=error_get_last())!==null) { | |
| 854 $this->clear(); | |
| 855 return false; | |
| 856 } | |
| 857 } | |
| 858 | |
| 859 // set callback function | |
| 860 function set_callback($function_name) { | |
| 861 $this->callback = $function_name; | |
| 862 } | |
| 863 | |
| 864 // remove callback function | |
| 865 function remove_callback() { | |
| 866 $this->callback = null; | |
| 867 } | |
| 868 | |
| 869 // save dom as string | |
| 870 function save($filepath='') { | |
| 871 $ret = $this->root->innertext(); | |
| 872 if ($filepath!=='') file_put_contents($filepath, $ret, LOCK_EX); | |
| 873 return $ret; | |
| 874 } | |
| 875 | |
| 876 // find dom node by css selector | |
| 877 // Paperg - allow us to specify that we want case insensitive testing of the value of the selector. | |
| 878 function find($selector, $idx=null, $lowercase=false) { | |
| 879 return $this->root->find($selector, $idx, $lowercase); | |
| 880 } | |
| 881 | |
| 882 // clean up memory due to php5 circular references memory leak... | |
| 883 function clear() { | |
| 884 foreach ($this->nodes as $n) {$n->clear(); $n = null;} | |
| 885 // This add next line is documented in the sourceforge repository. 2977248 as a fix for ongoing memory leaks that occur even with the use of clear. | |
| 886 if (isset($this->children)) foreach ($this->children as $n) {$n->clear(); $n = null;} | |
| 887 if (isset($this->parent)) {$this->parent->clear(); unset($this->parent);} | |
| 888 if (isset($this->root)) {$this->root->clear(); unset($this->root);} | |
| 889 unset($this->doc); | |
| 890 unset($this->noise); | |
| 891 } | |
| 892 | |
| 893 function dump($show_attr=true) { | |
| 894 $this->root->dump($show_attr); | |
| 895 } | |
| 896 | |
| 897 // prepare HTML data and init everything | |
| 898 protected function prepare($str, $lowercase=true, $stripRN=true, $defaultBRText=DEFAULT_BR_TEXT) { | |
| 899 $this->clear(); | |
| 900 | |
| 901 // set the length of content before we do anything to it. | |
| 902 $this->size = strlen($str); | |
| 903 | |
| 904 //before we save the string as the doc... strip out the \r \n's if we are told to. | |
| 905 if ($stripRN) { | |
| 906 $str = str_replace("\r", " ", $str); | |
| 907 $str = str_replace("\n", " ", $str); | |
| 908 } | |
| 909 | |
| 910 $this->doc = $str; | |
| 911 $this->pos = 0; | |
| 912 $this->cursor = 1; | |
| 913 $this->noise = array(); | |
| 914 $this->nodes = array(); | |
| 915 $this->lowercase = $lowercase; | |
| 916 $this->default_br_text = $defaultBRText; | |
| 917 $this->root = new simple_html_dom_node($this); | |
| 918 $this->root->tag = 'root'; | |
| 919 $this->root->_[HDOM_INFO_BEGIN] = -1; | |
| 920 $this->root->nodetype = HDOM_TYPE_ROOT; | |
| 921 $this->parent = $this->root; | |
| 922 if ($this->size>0) $this->char = $this->doc[0]; | |
| 923 } | |
| 924 | |
| 925 // parse html content | |
| 926 protected function parse() { | |
| 927 if (($s = $this->copy_until_char('<'))==='') | |
| 928 return $this->read_tag(); | |
| 929 | |
| 930 // text | |
| 931 $node = new simple_html_dom_node($this); | |
| 932 ++$this->cursor; | |
| 933 $node->_[HDOM_INFO_TEXT] = $s; | |
| 934 $this->link_nodes($node, false); | |
| 935 return true; | |
| 936 } | |
| 937 | |
| 938 // PAPERG - dkchou - added this to try to identify the character set of the page we have just parsed so we know better how to spit it out later. | |
| 939 // NOTE: IF you provide a routine called get_last_retrieve_url_contents_content_type which returns the CURLINFO_CONTENT_TYPE fromt he last curl_exec | |
| 940 // (or the content_type header fromt eh last transfer), we will parse THAT, and if a charset is specified, we will use it over any other mechanism. | |
| 941 protected function parse_charset() | |
| 942 { | |
| 943 global $debugObject; | |
| 944 | |
| 945 $charset = null; | |
| 946 | |
| 947 if (function_exists('get_last_retrieve_url_contents_content_type')) | |
| 948 { | |
| 949 $contentTypeHeader = get_last_retrieve_url_contents_content_type(); | |
| 950 $success = preg_match('/charset=(.+)/', $contentTypeHeader, $matches); | |
| 951 if ($success) | |
| 952 { | |
| 953 $charset = $matches[1]; | |
| 954 if (is_object($debugObject)) {$debugObject->debugLog(2, 'header content-type found charset of: ' . $charset);} | |
| 955 } | |
| 956 | |
| 957 } | |
| 958 | |
| 959 if (empty($charset)) | |
| 960 { | |
| 961 $el = $this->root->find('meta[http-equiv=Content-Type]',0); | |
| 962 if (!empty($el)) | |
| 963 { | |
| 964 $fullvalue = $el->content; | |
| 965 if (is_object($debugObject)) {$debugObject->debugLog(2, 'meta content-type tag found' . $fullValue);} | |
| 966 | |
| 967 if (!empty($fullvalue)) | |
| 968 { | |
| 969 $success = preg_match('/charset=(.+)/', $fullvalue, $matches); | |
| 970 if ($success) | |
| 971 { | |
| 972 $charset = $matches[1]; | |
| 973 } | |
| 974 else | |
| 975 { | |
| 976 // If there is a meta tag, and they don't specify the character set, research says that it's typically ISO-8859-1 | |
| 977 if (is_object($debugObject)) {$debugObject->debugLog(2, 'meta content-type tag couldn\'t be parsed. using iso-8859 default.');} | |
| 978 $charset = 'ISO-8859-1'; | |
| 979 } | |
| 980 } | |
| 981 } | |
| 982 } | |
| 983 | |
| 984 // If we couldn't find a charset above, then lets try to detect one based on the text we got... | |
| 985 if (empty($charset)) | |
| 986 { | |
| 987 // Have php try to detect the encoding from the text given to us. | |
| 988 $charset = mb_detect_encoding($this->root->plaintext . "ascii", $encoding_list = array( "UTF-8", "CP1252" ) ); | |
| 989 if (is_object($debugObject)) {$debugObject->debugLog(2, 'mb_detect found: ' . $charset);} | |
| 990 | |
| 991 // and if this doesn't work... then we need to just wrongheadedly assume it's UTF-8 so that we can move on - cause this will usually give us most of what we need... | |
| 992 if ($charset === false) | |
| 993 { | |
| 994 if (is_object($debugObject)) {$debugObject->debugLog(2, 'since mb_detect failed - using default of utf-8');} | |
| 995 $charset = 'UTF-8'; | |
| 996 } | |
| 997 } | |
| 998 | |
| 999 // Since CP1252 is a superset, if we get one of it's subsets, we want it instead. | |
| 1000 if ((strtolower($charset) == strtolower('ISO-8859-1')) || (strtolower($charset) == strtolower('Latin1')) || (strtolower($charset) == strtolower('Latin-1'))) | |
| 1001 { | |
| 1002 if (is_object($debugObject)) {$debugObject->debugLog(2, 'replacing ' . $charset . ' with CP1252 as its a superset');} | |
| 1003 $charset = 'CP1252'; | |
| 1004 } | |
| 1005 | |
| 1006 if (is_object($debugObject)) {$debugObject->debugLog(1, 'EXIT - ' . $charset);} | |
| 1007 | |
| 1008 return $this->_charset = $charset; | |
| 1009 } | |
| 1010 | |
| 1011 // read tag info | |
| 1012 protected function read_tag() { | |
| 1013 if ($this->char!=='<') { | |
| 1014 $this->root->_[HDOM_INFO_END] = $this->cursor; | |
| 1015 return false; | |
| 1016 } | |
| 1017 $begin_tag_pos = $this->pos; | |
| 1018 $this->char = (++$this->pos<$this->size) ? $this->doc[$this->pos] : null; // next | |
| 1019 | |
| 1020 // end tag | |
| 1021 if ($this->char==='/') { | |
| 1022 $this->char = (++$this->pos<$this->size) ? $this->doc[$this->pos] : null; // next | |
| 1023 // This represetns the change in the simple_html_dom trunk from revision 180 to 181. | |
| 1024 // $this->skip($this->token_blank_t); | |
| 1025 $this->skip($this->token_blank); | |
| 1026 $tag = $this->copy_until_char('>'); | |
| 1027 | |
| 1028 // skip attributes in end tag | |
| 1029 if (($pos = strpos($tag, ' '))!==false) | |
| 1030 $tag = substr($tag, 0, $pos); | |
| 1031 | |
| 1032 $parent_lower = strtolower($this->parent->tag); | |
| 1033 $tag_lower = strtolower($tag); | |
| 1034 | |
| 1035 if ($parent_lower!==$tag_lower) { | |
| 1036 if (isset($this->optional_closing_tags[$parent_lower]) && isset($this->block_tags[$tag_lower])) { | |
| 1037 $this->parent->_[HDOM_INFO_END] = 0; | |
| 1038 $org_parent = $this->parent; | |
| 1039 | |
| 1040 while (($this->parent->parent) && strtolower($this->parent->tag)!==$tag_lower) | |
| 1041 $this->parent = $this->parent->parent; | |
| 1042 | |
| 1043 if (strtolower($this->parent->tag)!==$tag_lower) { | |
| 1044 $this->parent = $org_parent; // restore origonal parent | |
| 1045 if ($this->parent->parent) $this->parent = $this->parent->parent; | |
| 1046 $this->parent->_[HDOM_INFO_END] = $this->cursor; | |
| 1047 return $this->as_text_node($tag); | |
| 1048 } | |
| 1049 } | |
| 1050 else if (($this->parent->parent) && isset($this->block_tags[$tag_lower])) { | |
| 1051 $this->parent->_[HDOM_INFO_END] = 0; | |
| 1052 $org_parent = $this->parent; | |
| 1053 | |
| 1054 while (($this->parent->parent) && strtolower($this->parent->tag)!==$tag_lower) | |
| 1055 $this->parent = $this->parent->parent; | |
| 1056 | |
| 1057 if (strtolower($this->parent->tag)!==$tag_lower) { | |
| 1058 $this->parent = $org_parent; // restore origonal parent | |
| 1059 $this->parent->_[HDOM_INFO_END] = $this->cursor; | |
| 1060 return $this->as_text_node($tag); | |
| 1061 } | |
| 1062 } | |
| 1063 else if (($this->parent->parent) && strtolower($this->parent->parent->tag)===$tag_lower) { | |
| 1064 $this->parent->_[HDOM_INFO_END] = 0; | |
| 1065 $this->parent = $this->parent->parent; | |
| 1066 } | |
| 1067 else | |
| 1068 return $this->as_text_node($tag); | |
| 1069 } | |
| 1070 | |
| 1071 $this->parent->_[HDOM_INFO_END] = $this->cursor; | |
| 1072 if ($this->parent->parent) $this->parent = $this->parent->parent; | |
| 1073 | |
| 1074 $this->char = (++$this->pos<$this->size) ? $this->doc[$this->pos] : null; // next | |
| 1075 return true; | |
| 1076 } | |
| 1077 | |
| 1078 $node = new simple_html_dom_node($this); | |
| 1079 $node->_[HDOM_INFO_BEGIN] = $this->cursor; | |
| 1080 ++$this->cursor; | |
| 1081 $tag = $this->copy_until($this->token_slash); | |
| 1082 $node->tag_start = $begin_tag_pos; | |
| 1083 | |
| 1084 // doctype, cdata & comments... | |
| 1085 if (isset($tag[0]) && $tag[0]==='!') { | |
| 1086 $node->_[HDOM_INFO_TEXT] = '<' . $tag . $this->copy_until_char('>'); | |
| 1087 | |
| 1088 if (isset($tag[2]) && $tag[1]==='-' && $tag[2]==='-') { | |
| 1089 $node->nodetype = HDOM_TYPE_COMMENT; | |
| 1090 $node->tag = 'comment'; | |
| 1091 } else { | |
| 1092 $node->nodetype = HDOM_TYPE_UNKNOWN; | |
| 1093 $node->tag = 'unknown'; | |
| 1094 } | |
| 1095 if ($this->char==='>') $node->_[HDOM_INFO_TEXT].='>'; | |
| 1096 $this->link_nodes($node, true); | |
| 1097 $this->char = (++$this->pos<$this->size) ? $this->doc[$this->pos] : null; // next | |
| 1098 return true; | |
| 1099 } | |
| 1100 | |
| 1101 // text | |
| 1102 if ($pos=strpos($tag, '<')!==false) { | |
| 1103 $tag = '<' . substr($tag, 0, -1); | |
| 1104 $node->_[HDOM_INFO_TEXT] = $tag; | |
| 1105 $this->link_nodes($node, false); | |
| 1106 $this->char = $this->doc[--$this->pos]; // prev | |
| 1107 return true; | |
| 1108 } | |
| 1109 | |
| 1110 if (!preg_match("/^[\w-:]+$/", $tag)) { | |
| 1111 $node->_[HDOM_INFO_TEXT] = '<' . $tag . $this->copy_until('<>'); | |
| 1112 if ($this->char==='<') { | |
| 1113 $this->link_nodes($node, false); | |
| 1114 return true; | |
| 1115 } | |
| 1116 | |
| 1117 if ($this->char==='>') $node->_[HDOM_INFO_TEXT].='>'; | |
| 1118 $this->link_nodes($node, false); | |
| 1119 $this->char = (++$this->pos<$this->size) ? $this->doc[$this->pos] : null; // next | |
| 1120 return true; | |
| 1121 } | |
| 1122 | |
| 1123 // begin tag | |
| 1124 $node->nodetype = HDOM_TYPE_ELEMENT; | |
| 1125 $tag_lower = strtolower($tag); | |
| 1126 $node->tag = ($this->lowercase) ? $tag_lower : $tag; | |
| 1127 | |
| 1128 // handle optional closing tags | |
| 1129 if (isset($this->optional_closing_tags[$tag_lower]) ) { | |
| 1130 while (isset($this->optional_closing_tags[$tag_lower][strtolower($this->parent->tag)])) { | |
| 1131 $this->parent->_[HDOM_INFO_END] = 0; | |
| 1132 $this->parent = $this->parent->parent; | |
| 1133 } | |
| 1134 $node->parent = $this->parent; | |
| 1135 } | |
| 1136 | |
| 1137 $guard = 0; // prevent infinity loop | |
| 1138 $space = array($this->copy_skip($this->token_blank), '', ''); | |
| 1139 | |
| 1140 // attributes | |
| 1141 do | |
| 1142 { | |
| 1143 if ($this->char!==null && $space[0]==='') break; | |
| 1144 $name = $this->copy_until($this->token_equal); | |
| 1145 if ($guard===$this->pos) { | |
| 1146 $this->char = (++$this->pos<$this->size) ? $this->doc[$this->pos] : null; // next | |
| 1147 continue; | |
| 1148 } | |
| 1149 $guard = $this->pos; | |
| 1150 | |
| 1151 // handle endless '<' | |
| 1152 if ($this->pos>=$this->size-1 && $this->char!=='>') { | |
| 1153 $node->nodetype = HDOM_TYPE_TEXT; | |
| 1154 $node->_[HDOM_INFO_END] = 0; | |
| 1155 $node->_[HDOM_INFO_TEXT] = '<'.$tag . $space[0] . $name; | |
| 1156 $node->tag = 'text'; | |
| 1157 $this->link_nodes($node, false); | |
| 1158 return true; | |
| 1159 } | |
| 1160 | |
| 1161 // handle mismatch '<' | |
| 1162 if ($this->doc[$this->pos-1]=='<') { | |
| 1163 $node->nodetype = HDOM_TYPE_TEXT; | |
| 1164 $node->tag = 'text'; | |
| 1165 $node->attr = array(); | |
| 1166 $node->_[HDOM_INFO_END] = 0; | |
| 1167 $node->_[HDOM_INFO_TEXT] = substr($this->doc, $begin_tag_pos, $this->pos-$begin_tag_pos-1); | |
| 1168 $this->pos -= 2; | |
| 1169 $this->char = (++$this->pos<$this->size) ? $this->doc[$this->pos] : null; // next | |
| 1170 $this->link_nodes($node, false); | |
| 1171 return true; | |
| 1172 } | |
| 1173 | |
| 1174 if ($name!=='/' && $name!=='') { | |
| 1175 $space[1] = $this->copy_skip($this->token_blank); | |
| 1176 $name = $this->restore_noise($name); | |
| 1177 if ($this->lowercase) $name = strtolower($name); | |
| 1178 if ($this->char==='=') { | |
| 1179 $this->char = (++$this->pos<$this->size) ? $this->doc[$this->pos] : null; // next | |
| 1180 $this->parse_attr($node, $name, $space); | |
| 1181 } | |
| 1182 else { | |
| 1183 //no value attr: nowrap, checked selected... | |
| 1184 $node->_[HDOM_INFO_QUOTE][] = HDOM_QUOTE_NO; | |
| 1185 $node->attr[$name] = true; | |
| 1186 if ($this->char!='>') $this->char = $this->doc[--$this->pos]; // prev | |
| 1187 } | |
| 1188 $node->_[HDOM_INFO_SPACE][] = $space; | |
| 1189 $space = array($this->copy_skip($this->token_blank), '', ''); | |
| 1190 } | |
| 1191 else | |
| 1192 break; | |
| 1193 } while ($this->char!=='>' && $this->char!=='/'); | |
| 1194 | |
| 1195 $this->link_nodes($node, true); | |
| 1196 $node->_[HDOM_INFO_ENDSPACE] = $space[0]; | |
| 1197 | |
| 1198 // check self closing | |
| 1199 if ($this->copy_until_char_escape('>')==='/') { | |
| 1200 $node->_[HDOM_INFO_ENDSPACE] .= '/'; | |
| 1201 $node->_[HDOM_INFO_END] = 0; | |
| 1202 } | |
| 1203 else { | |
| 1204 // reset parent | |
| 1205 if (!isset($this->self_closing_tags[strtolower($node->tag)])) $this->parent = $node; | |
| 1206 } | |
| 1207 $this->char = (++$this->pos<$this->size) ? $this->doc[$this->pos] : null; // next | |
| 1208 | |
| 1209 // If it's a BR tag, we need to set it's text to the default text. | |
| 1210 // This way when we see it in plaintext, we can generate formatting that the user wants. | |
| 1211 if ($node->tag == "br") { | |
| 1212 $node->_[HDOM_INFO_INNER] = $this->default_br_text; | |
| 1213 } | |
| 1214 | |
| 1215 return true; | |
| 1216 } | |
| 1217 | |
| 1218 // parse attributes | |
| 1219 protected function parse_attr($node, $name, &$space) { | |
| 1220 // Per sourceforge: http://sourceforge.net/tracker/?func=detail&aid=3061408&group_id=218559&atid=1044037 | |
| 1221 // If the attribute is already defined inside a tag, only pay atetntion to the first one as opposed to the last one. | |
| 1222 if (isset($node->attr[$name])) | |
| 1223 { | |
| 1224 return; | |
| 1225 } | |
| 1226 | |
| 1227 $space[2] = $this->copy_skip($this->token_blank); | |
| 1228 switch ($this->char) { | |
| 1229 case '"': | |
| 1230 $node->_[HDOM_INFO_QUOTE][] = HDOM_QUOTE_DOUBLE; | |
| 1231 $this->char = (++$this->pos<$this->size) ? $this->doc[$this->pos] : null; // next | |
| 1232 $node->attr[$name] = $this->restore_noise($this->copy_until_char_escape('"')); | |
| 1233 $this->char = (++$this->pos<$this->size) ? $this->doc[$this->pos] : null; // next | |
| 1234 break; | |
| 1235 case '\'': | |
| 1236 $node->_[HDOM_INFO_QUOTE][] = HDOM_QUOTE_SINGLE; | |
| 1237 $this->char = (++$this->pos<$this->size) ? $this->doc[$this->pos] : null; // next | |
| 1238 $node->attr[$name] = $this->restore_noise($this->copy_until_char_escape('\'')); | |
| 1239 $this->char = (++$this->pos<$this->size) ? $this->doc[$this->pos] : null; // next | |
| 1240 break; | |
| 1241 default: | |
| 1242 $node->_[HDOM_INFO_QUOTE][] = HDOM_QUOTE_NO; | |
| 1243 $node->attr[$name] = $this->restore_noise($this->copy_until($this->token_attr)); | |
| 1244 } | |
| 1245 // PaperG: Attributes should not have \r or \n in them, that counts as html whitespace. | |
| 1246 $node->attr[$name] = str_replace("\r", "", $node->attr[$name]); | |
| 1247 $node->attr[$name] = str_replace("\n", "", $node->attr[$name]); | |
| 1248 // PaperG: If this is a "class" selector, lets get rid of the preceeding and trailing space since some people leave it in the multi class case. | |
| 1249 if ($name == "class") { | |
| 1250 $node->attr[$name] = trim($node->attr[$name]); | |
| 1251 } | |
| 1252 } | |
| 1253 | |
| 1254 // link node's parent | |
| 1255 protected function link_nodes(&$node, $is_child) { | |
| 1256 $node->parent = $this->parent; | |
| 1257 $this->parent->nodes[] = $node; | |
| 1258 if ($is_child) | |
| 1259 $this->parent->children[] = $node; | |
| 1260 } | |
| 1261 | |
| 1262 // as a text node | |
| 1263 protected function as_text_node($tag) { | |
| 1264 $node = new simple_html_dom_node($this); | |
| 1265 ++$this->cursor; | |
| 1266 $node->_[HDOM_INFO_TEXT] = '</' . $tag . '>'; | |
| 1267 $this->link_nodes($node, false); | |
| 1268 $this->char = (++$this->pos<$this->size) ? $this->doc[$this->pos] : null; // next | |
| 1269 return true; | |
| 1270 } | |
| 1271 | |
| 1272 protected function skip($chars) { | |
| 1273 $this->pos += strspn($this->doc, $chars, $this->pos); | |
| 1274 $this->char = ($this->pos<$this->size) ? $this->doc[$this->pos] : null; // next | |
| 1275 } | |
| 1276 | |
| 1277 protected function copy_skip($chars) { | |
| 1278 $pos = $this->pos; | |
| 1279 $len = strspn($this->doc, $chars, $pos); | |
| 1280 $this->pos += $len; | |
| 1281 $this->char = ($this->pos<$this->size) ? $this->doc[$this->pos] : null; // next | |
| 1282 if ($len===0) return ''; | |
| 1283 return substr($this->doc, $pos, $len); | |
| 1284 } | |
| 1285 | |
| 1286 protected function copy_until($chars) { | |
| 1287 $pos = $this->pos; | |
| 1288 $len = strcspn($this->doc, $chars, $pos); | |
| 1289 $this->pos += $len; | |
| 1290 $this->char = ($this->pos<$this->size) ? $this->doc[$this->pos] : null; // next | |
| 1291 return substr($this->doc, $pos, $len); | |
| 1292 } | |
| 1293 | |
| 1294 protected function copy_until_char($char) { | |
| 1295 if ($this->char===null) return ''; | |
| 1296 | |
| 1297 if (($pos = strpos($this->doc, $char, $this->pos))===false) { | |
| 1298 $ret = substr($this->doc, $this->pos, $this->size-$this->pos); | |
| 1299 $this->char = null; | |
| 1300 $this->pos = $this->size; | |
| 1301 return $ret; | |
| 1302 } | |
| 1303 | |
| 1304 if ($pos===$this->pos) return ''; | |
| 1305 $pos_old = $this->pos; | |
| 1306 $this->char = $this->doc[$pos]; | |
| 1307 $this->pos = $pos; | |
| 1308 return substr($this->doc, $pos_old, $pos-$pos_old); | |
| 1309 } | |
| 1310 | |
| 1311 protected function copy_until_char_escape($char) { | |
| 1312 if ($this->char===null) return ''; | |
| 1313 | |
| 1314 $start = $this->pos; | |
| 1315 while (1) { | |
| 1316 if (($pos = strpos($this->doc, $char, $start))===false) { | |
| 1317 $ret = substr($this->doc, $this->pos, $this->size-$this->pos); | |
| 1318 $this->char = null; | |
| 1319 $this->pos = $this->size; | |
| 1320 return $ret; | |
| 1321 } | |
| 1322 | |
| 1323 if ($pos===$this->pos) return ''; | |
| 1324 | |
| 1325 if ($this->doc[$pos-1]==='\\') { | |
| 1326 $start = $pos+1; | |
| 1327 continue; | |
| 1328 } | |
| 1329 | |
| 1330 $pos_old = $this->pos; | |
| 1331 $this->char = $this->doc[$pos]; | |
| 1332 $this->pos = $pos; | |
| 1333 return substr($this->doc, $pos_old, $pos-$pos_old); | |
| 1334 } | |
| 1335 } | |
| 1336 | |
| 1337 // remove noise from html content | |
| 1338 protected function remove_noise($pattern, $remove_tag=false) { | |
| 1339 $count = preg_match_all($pattern, $this->doc, $matches, PREG_SET_ORDER|PREG_OFFSET_CAPTURE); | |
| 1340 | |
| 1341 for ($i=$count-1; $i>-1; --$i) { | |
| 1342 $key = '___noise___'.sprintf('% 3d', count($this->noise)+100); | |
| 1343 $idx = ($remove_tag) ? 0 : 1; | |
| 1344 $this->noise[$key] = $matches[$i][$idx][0]; | |
| 1345 $this->doc = substr_replace($this->doc, $key, $matches[$i][$idx][1], strlen($matches[$i][$idx][0])); | |
| 1346 } | |
| 1347 | |
| 1348 // reset the length of content | |
| 1349 $this->size = strlen($this->doc); | |
| 1350 if ($this->size>0) $this->char = $this->doc[0]; | |
| 1351 } | |
| 1352 | |
| 1353 // restore noise to html content | |
| 1354 function restore_noise($text) { | |
| 1355 while (($pos=strpos($text, '___noise___'))!==false) { | |
| 1356 $key = '___noise___'.$text[$pos+11].$text[$pos+12].$text[$pos+13]; | |
| 1357 if (isset($this->noise[$key])) | |
| 1358 $text = substr($text, 0, $pos).$this->noise[$key].substr($text, $pos+14); | |
| 1359 } | |
| 1360 return $text; | |
| 1361 } | |
| 1362 | |
| 1363 function __toString() { | |
| 1364 return $this->root->innertext(); | |
| 1365 } | |
| 1366 | |
| 1367 function __get($name) { | |
| 1368 switch ($name) { | |
| 1369 case 'outertext': | |
| 1370 return $this->root->innertext(); | |
| 1371 case 'innertext': | |
| 1372 return $this->root->innertext(); | |
| 1373 case 'plaintext': | |
| 1374 return $this->root->text(); | |
| 1375 case 'charset': | |
| 1376 return $this->_charset; | |
| 1377 case 'target_charset': | |
| 1378 return $this->_target_charset; | |
| 1379 } | |
| 1380 } | |
| 1381 | |
| 1382 // camel naming conventions | |
| 1383 function childNodes($idx=-1) {return $this->root->childNodes($idx);} | |
| 1384 function firstChild() {return $this->root->first_child();} | |
| 1385 function lastChild() {return $this->root->last_child();} | |
| 1386 function getElementById($id) {return $this->find("#$id", 0);} | |
| 1387 function getElementsById($id, $idx=null) {return $this->find("#$id", $idx);} | |
| 1388 function getElementByTagName($name) {return $this->find($name, 0);} | |
| 1389 function getElementsByTagName($name, $idx=-1) {return $this->find($name, $idx);} | |
| 1390 function loadFile() {$args = func_get_args();$this->load_file($args);} | |
| 1391 } | |
| 1392 | |
| 1393 ?> |
