Version Description
= * Added filter hook wpel_internal_link * Fixed use_js option bug * Fixed bug loading non-existing stylesheet * Minified javascripts
Download this release
Release Info
| Developer | freelancephp |
| Plugin | |
| Version | 1.52 |
| Comparing to | |
| See all releases | |
Code changes from version 1.51 to 1.52
- css/admin-wp-external-links.css +0 -37
- includes/class-admin-external-links.php +658 -638
- includes/class-wp-external-links.php +603 -593
- includes/phpQuery.php +5701 -5701
- includes/wp-plugin-dev-classes/class-wp-meta-box-page.php +663 -663
- includes/wp-plugin-dev-classes/class-wp-option-forms.php +359 -359
- js/admin-wp-external-links.js +2 -172
- js/src/admin-wp-external-links.js +173 -0
- js/src/wp-external-links.js +75 -0
- js/src/wp-option-forms.js +55 -0
- js/wp-external-links.js +2 -72
- js/wp-option-forms.js +0 -55
- readme.txt +212 -200
- wp-external-links.php +52 -52
css/admin-wp-external-links.css
DELETED
|
@@ -1,37 +0,0 @@
|
|
| 1 |
-
/* Admin - WP External Links Plugin */
|
| 2 |
-
.tooltip-help { text-decoration: none; }
|
| 3 |
-
|
| 4 |
-
/* Tipsy Plugin */
|
| 5 |
-
.tipsy {
|
| 6 |
-
padding: 5px;
|
| 7 |
-
/*
|
| 8 |
-
opacity: 0.9;
|
| 9 |
-
filter: alpha(opacity=90);
|
| 10 |
-
background-repeat: no-repeat;
|
| 11 |
-
background-image: url(../images/tipsy.gif);
|
| 12 |
-
*/
|
| 13 |
-
}
|
| 14 |
-
.tipsy-inner {
|
| 15 |
-
padding: 5px 8px 4px 8px;
|
| 16 |
-
color: white;
|
| 17 |
-
max-width: 200px;
|
| 18 |
-
text-align: center;
|
| 19 |
-
text-shadow: 0 -1px 0 #333;
|
| 20 |
-
/* background-position: top right;*/
|
| 21 |
-
border-top:1px solid #808080;
|
| 22 |
-
border-botom:1px solid #6d6d6d;
|
| 23 |
-
-webkit-border-radius: 3px;
|
| 24 |
-
-moz-border-radius: 3px;
|
| 25 |
-
border-radius: 3px;
|
| 26 |
-
background-color:#777;
|
| 27 |
-
background-image:-ms-linear-gradient(bottom,#6d6d6d,#808080);
|
| 28 |
-
background-image:-moz-linear-gradient(bottom,#6d6d6d,#808080);
|
| 29 |
-
background-image:-o-linear-gradient(bottom,#6d6d6d,#808080);
|
| 30 |
-
background-image:-webkit-gradient(linear,left bottom,left top,from(#6d6d6d),to(#808080));
|
| 31 |
-
background-image:-webkit-linear-gradient(bottom,#6d6d6d,#808080);
|
| 32 |
-
background-image:linear-gradient(bottom,#6d6d6d,#808080);
|
| 33 |
-
}
|
| 34 |
-
.tipsy-north { background-position: top center; }
|
| 35 |
-
.tipsy-south { background-position: bottom center; }
|
| 36 |
-
.tipsy-east { background-position: right center; }
|
| 37 |
-
.tipsy-west { background-position: center bottom; }
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
includes/class-admin-external-links.php
CHANGED
|
@@ -1,639 +1,659 @@
|
|
| 1 |
-
<?php defined( 'ABSPATH' ) OR die( 'No direct access.' );
|
| 2 |
-
if ( ! class_exists( 'Admin_External_Links' ) ):
|
| 3 |
-
|
| 4 |
-
/**
|
| 5 |
-
* Class Admin_External_Links
|
| 6 |
-
* @category WordPress Plugins
|
| 7 |
-
*/
|
| 8 |
-
final class Admin_External_Links {
|
| 9 |
-
|
| 10 |
-
/**
|
| 11 |
-
* Options to be saved and their default values
|
| 12 |
-
* @var array
|
| 13 |
-
*/
|
| 14 |
-
public $save_options = array(
|
| 15 |
-
'meta' => array(
|
| 16 |
-
'version' => NULL,
|
| 17 |
-
),
|
| 18 |
-
'main' => array(
|
| 19 |
-
'target' => '_none',
|
| 20 |
-
'
|
| 21 |
-
'
|
| 22 |
-
'
|
| 23 |
-
'
|
| 24 |
-
'
|
| 25 |
-
|
| 26 |
-
|
| 27 |
-
|
| 28 |
-
'
|
| 29 |
-
'
|
| 30 |
-
'
|
| 31 |
-
'load_in_footer' => 1,
|
| 32 |
-
),
|
| 33 |
-
'style' => array(
|
| 34 |
-
'class_name' => 'ext-link',
|
| 35 |
-
'icon' => 0,
|
| 36 |
-
'image_no_icon' => 1,
|
| 37 |
-
'no_icon_class' => 'no-ext-icon',
|
| 38 |
-
'no_icon_same_window' => 0,
|
| 39 |
-
),
|
| 40 |
-
'extra' => array(
|
| 41 |
-
'fix_js' => 0,
|
| 42 |
-
'phpquery' => 0,
|
| 43 |
-
'filter_excl_sel' => '.excl-ext-link',
|
| 44 |
-
),
|
| 45 |
-
'screen' => array(
|
| 46 |
-
'menu_position' => NULL,
|
| 47 |
-
),
|
| 48 |
-
);
|
| 49 |
-
|
| 50 |
-
/**
|
| 51 |
-
* Meta box page object
|
| 52 |
-
* @var WP_Meta_Box_Page
|
| 53 |
-
*/
|
| 54 |
-
public $meta_box_page = NULL;
|
| 55 |
-
|
| 56 |
-
/**
|
| 57 |
-
* Ajax form object
|
| 58 |
-
* @var WP_Ajax_Option_Form
|
| 59 |
-
*/
|
| 60 |
-
public $form = NULL;
|
| 61 |
-
static public $staticForm = NULL;
|
| 62 |
-
|
| 63 |
-
|
| 64 |
-
/**
|
| 65 |
-
* Constructor
|
| 66 |
-
*/
|
| 67 |
-
public function __construct() {
|
| 68 |
-
$this->check_version_update();
|
| 69 |
-
|
| 70 |
-
// set meta box page
|
| 71 |
-
$this->meta_box_page = new WP_Meta_Box_Page_01();
|
| 72 |
-
|
| 73 |
-
// set ajax forms (also used by front-end)
|
| 74 |
-
$this->form = new WP_Option_Forms_01( WP_EXTERNAL_LINKS_KEY, $this->save_options );
|
| 75 |
-
self::$staticForm = $this->form;
|
| 76 |
-
|
| 77 |
-
// init admin
|
| 78 |
-
add_action( 'admin_init', array( $this, 'admin_init' ) );
|
| 79 |
-
|
| 80 |
-
if ( is_admin() ) {
|
| 81 |
-
// set options for add_page_method
|
| 82 |
-
$menu_pos = $this->form->set_current_option( 'screen' )->value( 'menu_position' );
|
| 83 |
-
|
| 84 |
-
// init meta box page
|
| 85 |
-
$this->meta_box_page->init(
|
| 86 |
-
// settings
|
| 87 |
-
array(
|
| 88 |
-
'page_title' => $this->__( 'WP External Links' ),
|
| 89 |
-
'menu_title' => $this->__( 'External Links' ),
|
| 90 |
-
'page_slug' => strtolower( WP_EXTERNAL_LINKS_KEY ),
|
| 91 |
-
'add_page_method' => ( ! empty( $menu_pos ) AND $menu_pos != 'admin.php' ) ? 'add_submenu_page' : 'add_menu_page',
|
| 92 |
-
'parent_slug' => ( ! empty( $menu_pos ) AND $menu_pos != 'admin.php' ) ? $menu_pos : NULL,
|
| 93 |
-
'column_widths' => array(
|
| 94 |
-
1 => array( 99 ),
|
| 95 |
-
2 => array( 69, 29 ),
|
| 96 |
-
),
|
| 97 |
-
'icon_url' => plugins_url( 'images/icon-wp-external-links-16.png', WP_EXTERNAL_LINKS_FILE ),
|
| 98 |
-
),
|
| 99 |
-
// load callback
|
| 100 |
-
array( $this, 'call_load_meta_box' )
|
| 101 |
-
);
|
| 102 |
-
}
|
| 103 |
-
}
|
| 104 |
-
|
| 105 |
-
/**
|
| 106 |
-
* Initialize Admin
|
| 107 |
-
*/
|
| 108 |
-
public function admin_init() {
|
| 109 |
-
// set uninstall hook
|
| 110 |
-
register_uninstall_hook( WP_EXTERNAL_LINKS_FILE, array( 'Admin_External_Links', 'call_uninstall' ) );
|
| 111 |
-
|
| 112 |
-
// load text domain for translations
|
| 113 |
-
load_plugin_textdomain( WP_EXTERNAL_LINKS_KEY, FALSE, dirname( plugin_basename( WP_EXTERNAL_LINKS_FILE ) ) . '/lang/' );
|
| 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 |
-
<br
|
| 225 |
-
|
| 226 |
-
|
| 227 |
-
|
| 228 |
-
<br
|
| 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 |
-
<?php
|
| 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 |
-
<tr>
|
| 371 |
-
<th
|
| 372 |
-
<?php echo $this->tooltip_help( '
|
| 373 |
-
<td><label><?php echo $this->form->
|
| 374 |
-
|
| 375 |
-
|
| 376 |
-
|
| 377 |
-
|
| 378 |
-
|
| 379 |
-
|
| 380 |
-
|
| 381 |
-
|
| 382 |
-
|
| 383 |
-
|
| 384 |
-
|
| 385 |
-
|
| 386 |
-
|
| 387 |
-
|
| 388 |
-
|
| 389 |
-
|
| 390 |
-
|
| 391 |
-
|
| 392 |
-
|
| 393 |
-
|
| 394 |
-
|
| 395 |
-
<?php
|
| 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 |
-
<?php
|
| 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 |
-
<p><?php
|
| 469 |
-
|
| 470 |
-
|
| 471 |
-
|
| 472 |
-
|
| 473 |
-
|
| 474 |
-
|
| 475 |
-
|
| 476 |
-
|
| 477 |
-
|
| 478 |
-
|
| 479 |
-
|
| 480 |
-
<?php
|
| 481 |
-
<p
|
| 482 |
-
<?php
|
| 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 |
-
$new_options
|
| 562 |
-
|
| 563 |
-
$
|
| 564 |
-
|
| 565 |
-
if ( isset( $general[ '
|
| 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 |
/* ommit PHP closing tag, to prevent unwanted whitespace at the end of the parts generated by the included files */
|
| 1 |
+
<?php defined( 'ABSPATH' ) OR die( 'No direct access.' );
|
| 2 |
+
if ( ! class_exists( 'Admin_External_Links' ) ):
|
| 3 |
+
|
| 4 |
+
/**
|
| 5 |
+
* Class Admin_External_Links
|
| 6 |
+
* @category WordPress Plugins
|
| 7 |
+
*/
|
| 8 |
+
final class Admin_External_Links {
|
| 9 |
+
|
| 10 |
+
/**
|
| 11 |
+
* Options to be saved and their default values
|
| 12 |
+
* @var array
|
| 13 |
+
*/
|
| 14 |
+
public $save_options = array(
|
| 15 |
+
'meta' => array(
|
| 16 |
+
'version' => NULL,
|
| 17 |
+
),
|
| 18 |
+
'main' => array(
|
| 19 |
+
'target' => '_none',
|
| 20 |
+
'filter_page' => 1,
|
| 21 |
+
'filter_posts' => 1,
|
| 22 |
+
'filter_comments' => 1,
|
| 23 |
+
'filter_widgets' => 1,
|
| 24 |
+
'ignore' => '//twitter.com/share',
|
| 25 |
+
),
|
| 26 |
+
'seo' => array(
|
| 27 |
+
'external' => 1,
|
| 28 |
+
'nofollow' => 1,
|
| 29 |
+
'title' => '%title%',
|
| 30 |
+
'use_js' => 1,
|
| 31 |
+
'load_in_footer' => 1,
|
| 32 |
+
),
|
| 33 |
+
'style' => array(
|
| 34 |
+
'class_name' => 'ext-link',
|
| 35 |
+
'icon' => 0,
|
| 36 |
+
'image_no_icon' => 1,
|
| 37 |
+
'no_icon_class' => 'no-ext-icon',
|
| 38 |
+
'no_icon_same_window' => 0,
|
| 39 |
+
),
|
| 40 |
+
'extra' => array(
|
| 41 |
+
'fix_js' => 0,
|
| 42 |
+
'phpquery' => 0,
|
| 43 |
+
'filter_excl_sel' => '.excl-ext-link',
|
| 44 |
+
),
|
| 45 |
+
'screen' => array(
|
| 46 |
+
'menu_position' => NULL,
|
| 47 |
+
),
|
| 48 |
+
);
|
| 49 |
+
|
| 50 |
+
/**
|
| 51 |
+
* Meta box page object
|
| 52 |
+
* @var WP_Meta_Box_Page
|
| 53 |
+
*/
|
| 54 |
+
public $meta_box_page = NULL;
|
| 55 |
+
|
| 56 |
+
/**
|
| 57 |
+
* Ajax form object
|
| 58 |
+
* @var WP_Ajax_Option_Form
|
| 59 |
+
*/
|
| 60 |
+
public $form = NULL;
|
| 61 |
+
static public $staticForm = NULL;
|
| 62 |
+
|
| 63 |
+
|
| 64 |
+
/**
|
| 65 |
+
* Constructor
|
| 66 |
+
*/
|
| 67 |
+
public function __construct() {
|
| 68 |
+
$this->check_version_update();
|
| 69 |
+
|
| 70 |
+
// set meta box page
|
| 71 |
+
$this->meta_box_page = new WP_Meta_Box_Page_01();
|
| 72 |
+
|
| 73 |
+
// set ajax forms (also used by front-end)
|
| 74 |
+
$this->form = new WP_Option_Forms_01( WP_EXTERNAL_LINKS_KEY, $this->save_options );
|
| 75 |
+
self::$staticForm = $this->form;
|
| 76 |
+
|
| 77 |
+
// init admin
|
| 78 |
+
add_action( 'admin_init', array( $this, 'admin_init' ) );
|
| 79 |
+
|
| 80 |
+
if ( is_admin() ) {
|
| 81 |
+
// set options for add_page_method
|
| 82 |
+
$menu_pos = $this->form->set_current_option( 'screen' )->value( 'menu_position' );
|
| 83 |
+
|
| 84 |
+
// init meta box page
|
| 85 |
+
$this->meta_box_page->init(
|
| 86 |
+
// settings
|
| 87 |
+
array(
|
| 88 |
+
'page_title' => $this->__( 'WP External Links' ),
|
| 89 |
+
'menu_title' => $this->__( 'External Links' ),
|
| 90 |
+
'page_slug' => strtolower( WP_EXTERNAL_LINKS_KEY ),
|
| 91 |
+
'add_page_method' => ( ! empty( $menu_pos ) AND $menu_pos != 'admin.php' ) ? 'add_submenu_page' : 'add_menu_page',
|
| 92 |
+
'parent_slug' => ( ! empty( $menu_pos ) AND $menu_pos != 'admin.php' ) ? $menu_pos : NULL,
|
| 93 |
+
'column_widths' => array(
|
| 94 |
+
1 => array( 99 ),
|
| 95 |
+
2 => array( 69, 29 ),
|
| 96 |
+
),
|
| 97 |
+
'icon_url' => plugins_url( 'images/icon-wp-external-links-16.png', WP_EXTERNAL_LINKS_FILE ),
|
| 98 |
+
),
|
| 99 |
+
// load callback
|
| 100 |
+
array( $this, 'call_load_meta_box' )
|
| 101 |
+
);
|
| 102 |
+
}
|
| 103 |
+
}
|
| 104 |
+
|
| 105 |
+
/**
|
| 106 |
+
* Initialize Admin
|
| 107 |
+
*/
|
| 108 |
+
public function admin_init() {
|
| 109 |
+
// set uninstall hook
|
| 110 |
+
register_uninstall_hook( WP_EXTERNAL_LINKS_FILE, array( 'Admin_External_Links', 'call_uninstall' ) );
|
| 111 |
+
|
| 112 |
+
// load text domain for translations
|
| 113 |
+
load_plugin_textdomain( WP_EXTERNAL_LINKS_KEY, FALSE, dirname( plugin_basename( WP_EXTERNAL_LINKS_FILE ) ) . '/lang/' );
|
| 114 |
+
}
|
| 115 |
+
|
| 116 |
+
/**
|
| 117 |
+
* Add to head of Admin page
|
| 118 |
+
*/
|
| 119 |
+
public function admin_head() {
|
| 120 |
+
echo <<< style
|
| 121 |
+
<style type="text/css">
|
| 122 |
+
/* WP External Links */
|
| 123 |
+
.postbox-container { margin-left:1%; }
|
| 124 |
+
.tooltip-help { text-decoration: none; }
|
| 125 |
+
.tipsy { padding: 5px; }
|
| 126 |
+
.tipsy-inner { padding: 5px 8px 4px 8px; color: white; max-width: 200px; text-align: center; text-shadow: 0 -1px 0 #333;
|
| 127 |
+
border-top:1px solid #808080; border-botom:1px solid #6d6d6d; -webkit-border-radius: 3px; -moz-border-radius: 3px; border-radius: 3px;
|
| 128 |
+
background-color:#777; background-image:-ms-linear-gradient(bottom,#6d6d6d,#808080); background-image:-moz-linear-gradient(bottom,#6d6d6d,#808080); background-image:-o-linear-gradient(bottom,#6d6d6d,#808080); background-image:-webkit-gradient(linear,left bottom,left top,from(#6d6d6d),to(#808080)); background-image:-webkit-linear-gradient(bottom,#6d6d6d,#808080); background-image:linear-gradient(bottom,#6d6d6d,#808080);
|
| 129 |
+
}
|
| 130 |
+
.tipsy-north { background-position: top center; }
|
| 131 |
+
.tipsy-south { background-position: bottom center; }
|
| 132 |
+
.tipsy-east { background-position: right center; }
|
| 133 |
+
.tipsy-west { background-position: center bottom; }
|
| 134 |
+
<style>
|
| 135 |
+
style;
|
| 136 |
+
}
|
| 137 |
+
|
| 138 |
+
/**
|
| 139 |
+
* Translate text in current domain
|
| 140 |
+
* @param string $text
|
| 141 |
+
* @return string
|
| 142 |
+
*/
|
| 143 |
+
public function __( $text ) {
|
| 144 |
+
return translate( $text, WP_EXTERNAL_LINKS_KEY );
|
| 145 |
+
}
|
| 146 |
+
|
| 147 |
+
/**
|
| 148 |
+
* Translate text in current domain
|
| 149 |
+
* @param string $text
|
| 150 |
+
* @return string
|
| 151 |
+
*/
|
| 152 |
+
public function _e( $text ) {
|
| 153 |
+
echo translate( $text, WP_EXTERNAL_LINKS_KEY );
|
| 154 |
+
}
|
| 155 |
+
|
| 156 |
+
/**
|
| 157 |
+
* Load meta box action
|
| 158 |
+
*/
|
| 159 |
+
public function call_load_meta_box( $meta_box ) {
|
| 160 |
+
add_action( 'admin_head', array($this, 'admin_head') );
|
| 161 |
+
|
| 162 |
+
// add filters
|
| 163 |
+
$meta_box->add_title_filter( array( $this, 'call_page_title' ) )
|
| 164 |
+
->add_contextual_help_filter( array( $this, 'call_contextual_help' ) );
|
| 165 |
+
|
| 166 |
+
// add meta boxes
|
| 167 |
+
// add_meta_box( $title, $callback, $context = 'normal', $id = NULL, $priority = 'default', $callback_args = NULL )
|
| 168 |
+
$meta_box->add_meta_box( $this->__( 'General Settings' ), array( $this, 'call_box_general_settings' ), 1 )
|
| 169 |
+
->add_meta_box( $this->__( 'SEO Settings' ), array( $this, 'call_box_seo_settings' ), 1 )
|
| 170 |
+
->add_meta_box( $this->__( 'Style Settings' ), array( $this, 'call_box_style_settings' ), 1 )
|
| 171 |
+
->add_meta_box( $this->__( 'Extra Settings' ), array( $this, 'call_box_extra_settings' ), 1 )
|
| 172 |
+
->add_meta_box( $this->__( 'Admin Settings' ), array( $this, 'call_box_admin_settings' ), 1 )
|
| 173 |
+
//->add_meta_box( $this->__( 'About this Plugin' ), array( $this, 'call_box_about' ), 2 )
|
| 174 |
+
->add_meta_box( $this->__( 'Other Plugins' ), array( $this, 'call_box_other_plugins' ), 2 );
|
| 175 |
+
|
| 176 |
+
// scripts
|
| 177 |
+
wp_enqueue_script( 'admin-wp-external-links', plugins_url( '/js/admin-wp-external-links.js', WP_EXTERNAL_LINKS_FILE ), array( 'postbox' ), WP_EXTERNAL_LINKS_VERSION );
|
| 178 |
+
}
|
| 179 |
+
|
| 180 |
+
/**
|
| 181 |
+
* Contextual_help (callback)
|
| 182 |
+
* @param string $content
|
| 183 |
+
* @return string
|
| 184 |
+
*/
|
| 185 |
+
public function call_contextual_help( $content ) {
|
| 186 |
+
$help = '';
|
| 187 |
+
$help .= $this->meta_box_page->get_ob_callback( array( $this, 'call_box_about' ) );
|
| 188 |
+
return $help . $content;
|
| 189 |
+
}
|
| 190 |
+
|
| 191 |
+
/**
|
| 192 |
+
* Add icon to page title
|
| 193 |
+
* @return string
|
| 194 |
+
*/
|
| 195 |
+
public function call_page_title( $title ) {
|
| 196 |
+
// when updated set the update message
|
| 197 |
+
if ( isset($_GET[ 'settings-updated' ]) && $_GET[ 'settings-updated' ] == 'true' ) {
|
| 198 |
+
$title .= '<div class="updated settings-error" id="setting-error-settings_updated">'
|
| 199 |
+
. '<p><strong>' . __( 'Settings saved.' ) .'</strong></p>'
|
| 200 |
+
. '</div>';
|
| 201 |
+
}
|
| 202 |
+
|
| 203 |
+
$title = '<div class="icon32" id="icon-options-custom" style="background:url( '. plugins_url( 'images/icon-wp-external-links-32.png', WP_EXTERNAL_LINKS_FILE ) .' ) no-repeat 50% 50%"><br></div>'
|
| 204 |
+
. $title;
|
| 205 |
+
|
| 206 |
+
return $title;
|
| 207 |
+
}
|
| 208 |
+
|
| 209 |
+
/**
|
| 210 |
+
* Meta Box: General Settings
|
| 211 |
+
*/
|
| 212 |
+
public function call_box_general_settings() {
|
| 213 |
+
echo $this->form->set_current_option( 'main' )->open_form();
|
| 214 |
+
?>
|
| 215 |
+
<fieldset class="options">
|
| 216 |
+
<table class="form-table">
|
| 217 |
+
<tr>
|
| 218 |
+
<th style="width:250px;"><?php $this->_e( 'Open external links in...' ) ?>
|
| 219 |
+
<?php echo $this->tooltip_help( 'Specify the target (window or tab) for opening external links.' ) ?></th>
|
| 220 |
+
<td class="target_external_links">
|
| 221 |
+
<label><?php echo $this->form->radio( 'target', '_none', array( 'class' => 'field_target' ) ); ?>
|
| 222 |
+
<span><?php $this->_e( 'Same window or tab (<code>_none</code>)' ) ?></span></label>
|
| 223 |
+
<?php echo $this->tooltip_help( 'Open in current window or tab, when framed in the same frame.' ) ?>
|
| 224 |
+
<br/>
|
| 225 |
+
<label><?php echo $this->form->radio( 'target', '_blank', array( 'class' => 'field_target' ) ); ?>
|
| 226 |
+
<span><?php $this->_e( 'New window or tab (<code>_blank</code>)' ) ?></span></label>
|
| 227 |
+
<?php echo $this->tooltip_help( 'Open every external link in a new window or tab.' ) ?>
|
| 228 |
+
<br style="margin-bottom:15px;" />
|
| 229 |
+
<label><?php echo $this->form->radio( 'target', '_top', array( 'class' => 'field_target' ) ); ?>
|
| 230 |
+
<span><?php $this->_e( 'Topmost frame (<code>_top</code>)' ) ?></span></label>
|
| 231 |
+
<?php echo $this->tooltip_help( 'Open in current window or tab, when framed in the topmost frame.' ) ?>
|
| 232 |
+
<br/>
|
| 233 |
+
<label><?php echo $this->form->radio( 'target', '_new', array( 'class' => 'field_target' ) ); ?>
|
| 234 |
+
<span><?php $this->_e( 'Seperate window or tab (<code>_new</code>)' ) ?></span></label>
|
| 235 |
+
<?php echo $this->tooltip_help( 'Open new window the first time and use this window for each external link.' ) ?>
|
| 236 |
+
</td>
|
| 237 |
+
</tr>
|
| 238 |
+
<tr>
|
| 239 |
+
<th style="width:250px;"><?php $this->_e( 'Apply plugin settings on...' ) ?>
|
| 240 |
+
<?php echo $this->tooltip_help( 'Choose contents for applying settings to external links.' ) ?></th>
|
| 241 |
+
<td>
|
| 242 |
+
<label><?php echo $this->form->checkbox( 'filter_page', 1 ); ?>
|
| 243 |
+
<span><?php $this->_e( 'All contents' ) ?></span> <span class="description"><?php $this->_e('(the whole <code><body></code>)') ?></span></label>
|
| 244 |
+
<br/> <label><?php echo $this->form->checkbox( 'filter_posts', 1 ); ?>
|
| 245 |
+
<span><?php $this->_e( 'Post contents' ) ?></span></label>
|
| 246 |
+
<br/> <label><?php echo $this->form->checkbox( 'filter_comments', 1 ); ?>
|
| 247 |
+
<span><?php $this->_e( 'Comments' ) ?></span></label>
|
| 248 |
+
<br/> <label><?php echo $this->form->checkbox( 'filter_widgets', 1 ); ?>
|
| 249 |
+
<span><?php
|
| 250 |
+
if ( self::check_widget_content_filter() ):
|
| 251 |
+
$this->_e( 'All widgets' );
|
| 252 |
+
echo $this->tooltip_help( 'Applied to all widgets by using the "widget_content" filter of the Widget Logic plugin' );
|
| 253 |
+
else:
|
| 254 |
+
$this->_e( 'All text widgets' );
|
| 255 |
+
echo $this->tooltip_help( 'Only the text widget will be applied. To apply to all widget you should select "All contents" option.' );
|
| 256 |
+
endif;
|
| 257 |
+
?></span></label>
|
| 258 |
+
</td>
|
| 259 |
+
</tr>
|
| 260 |
+
<tr>
|
| 261 |
+
<th><?php $this->_e( 'Ignore links (URL) containing...' ) ?>
|
| 262 |
+
<?php echo $this->tooltip_help( 'This plugin will completely ignore links that contain one of the given texts in the URL. Use enter to seperate each text. This check is not case sensitive.' ) ?></th>
|
| 263 |
+
<td><label><?php echo $this->form->textarea( 'ignore' ); ?>
|
| 264 |
+
<span class="description"><?php _e( 'Be as specific as you want, f.e.: <code>twitter.com</code> or <code>https://twitter.com</code>. Seperate each by an enter.' ) ?></span></label>
|
| 265 |
+
</td>
|
| 266 |
+
</tr>
|
| 267 |
+
</table>
|
| 268 |
+
</fieldset>
|
| 269 |
+
|
| 270 |
+
<?php
|
| 271 |
+
echo $this->form->submit();
|
| 272 |
+
echo $this->form->close_form();
|
| 273 |
+
}
|
| 274 |
+
|
| 275 |
+
/**
|
| 276 |
+
* Meta Box: SEO Settings
|
| 277 |
+
*/
|
| 278 |
+
public function call_box_seo_settings() {
|
| 279 |
+
echo $this->form->set_current_option( 'seo' )->open_form();
|
| 280 |
+
?>
|
| 281 |
+
<fieldset class="options">
|
| 282 |
+
<table class="form-table">
|
| 283 |
+
<tr>
|
| 284 |
+
<th style="width:250px;"><?php $this->_e( 'Add to <code>rel</code>-attribute' ) ?>
|
| 285 |
+
<?php echo $this->tooltip_help( 'Set values for the "rel"-atribute of external links.' ) ?></th>
|
| 286 |
+
<td><label><?php echo $this->form->checkbox( 'nofollow', 1 ); ?>
|
| 287 |
+
<span><?php $this->_e( 'Add <code>"nofollow"</code>' ) ?></span></label>
|
| 288 |
+
<?php echo $this->tooltip_help( 'Add "nofollow" to the "rel"-attribute of external links (unless link already has "follow").' ) ?>
|
| 289 |
+
<br/>
|
| 290 |
+
<label><?php echo $this->form->checkbox( 'external', 1 ); ?>
|
| 291 |
+
<span><?php $this->_e( 'Add <code>"external"</code>' ) ?></span></label>
|
| 292 |
+
<?php echo $this->tooltip_help( 'Add "external" to the "rel"-attribute of external links.' ) ?>
|
| 293 |
+
</td>
|
| 294 |
+
</tr>
|
| 295 |
+
<tr>
|
| 296 |
+
<th><?php $this->_e( 'Set <code>title</code>-attribute' ) ?>
|
| 297 |
+
<?php echo $this->tooltip_help( 'Set title attribute for external links. Use %title% for the original title value.' ) ?></th>
|
| 298 |
+
<td><label><?php echo $this->form->text( 'title' ); ?>
|
| 299 |
+
<br/><span class="description"><?php _e( 'Use <code>%title%</code> for the original title value.' ) ?></span></label></td>
|
| 300 |
+
</tr>
|
| 301 |
+
<tr>
|
| 302 |
+
<th><?php $this->_e( 'Use JavaScript method' ) ?>
|
| 303 |
+
<?php echo $this->tooltip_help( 'Enable this option to use the JavaScript method for opening links, which prevents adding target attribute in the HTML code.' ) ?></label>
|
| 304 |
+
</th>
|
| 305 |
+
<td>
|
| 306 |
+
<label><?php echo $this->form->checkbox( 'use_js', 1, array( 'class' => 'field_use_js' ) ); ?>
|
| 307 |
+
<span><?php $this->_e( 'Use JavaScript for opening links' ) ?></span> <span class="description"><?php $this->_e( '(valid xhtml strict)' ) ?></span>
|
| 308 |
+
<br/>
|
| 309 |
+
<label><?php echo $this->form->checkbox( 'load_in_footer', 1, array( 'class' => 'load_in_footer' ) ); ?>
|
| 310 |
+
<span><?php $this->_e( 'Load JS file in footer' ) ?></span>
|
| 311 |
+
</td>
|
| 312 |
+
</tr>
|
| 313 |
+
</table>
|
| 314 |
+
</fieldset>
|
| 315 |
+
<?php
|
| 316 |
+
echo $this->form->submit();
|
| 317 |
+
echo $this->form->close_form();
|
| 318 |
+
}
|
| 319 |
+
|
| 320 |
+
/**
|
| 321 |
+
* Meta Box: Style Settings
|
| 322 |
+
*/
|
| 323 |
+
public function call_box_style_settings() {
|
| 324 |
+
echo $this->form->set_current_option( 'style' )->open_form();
|
| 325 |
+
?>
|
| 326 |
+
<fieldset class="options">
|
| 327 |
+
<table class="form-table">
|
| 328 |
+
<tr>
|
| 329 |
+
<th style="width:250px;"><?php $this->_e( 'Set icon for external link' ) ?>
|
| 330 |
+
<?php echo $this->tooltip_help( 'Set an icon that wll be shown for external links. See example on the right side.' ) ?></th>
|
| 331 |
+
<td>
|
| 332 |
+
<div>
|
| 333 |
+
<div style="width:15%;float:left">
|
| 334 |
+
<label><?php echo $this->form->radio( 'icon', 0 ); ?>
|
| 335 |
+
<span><?php $this->_e( 'No icon' ) ?></span></label>
|
| 336 |
+
<?php for ( $x = 1; $x <= 20; $x++ ): ?>
|
| 337 |
+
<br/>
|
| 338 |
+
<label title="<?php echo sprintf( $this->__( 'Icon %1$s: choose this icon to show for all external links or add the class \'ext-icon-%1$s\' to a specific link.' ), $x ) ?>">
|
| 339 |
+
<?php echo $this->form->radio( 'icon', $x ); ?>
|
| 340 |
+
<img src="<?php echo plugins_url('images/ext-icons/ext-icon-'. $x .'.png', WP_EXTERNAL_LINKS_FILE) ?>" /></label>
|
| 341 |
+
<?php if ( $x % 5 == 0 ): ?>
|
| 342 |
+
</div>
|
| 343 |
+
<div style="width:15%;float:left">
|
| 344 |
+
<?php endif; ?>
|
| 345 |
+
<?php endfor; ?>
|
| 346 |
+
</div>
|
| 347 |
+
<div style="width:29%;float:left;"><span class="description"><?php $this->_e( 'Example:' ) ?></span>
|
| 348 |
+
<br/><img src="<?php echo plugins_url( 'images/link-icon-example.png', WP_EXTERNAL_LINKS_FILE ) ?>" />
|
| 349 |
+
</div>
|
| 350 |
+
<br style="clear:both" />
|
| 351 |
+
</div>
|
| 352 |
+
</td>
|
| 353 |
+
</tr>
|
| 354 |
+
<tr>
|
| 355 |
+
<th><?php $this->_e( 'Skip images' ) ?>
|
| 356 |
+
<?php echo $this->tooltip_help( 'Don\'t show icon for external links containing images.' ) ?></th>
|
| 357 |
+
<td><label><?php echo $this->form->checkbox( 'image_no_icon', 1 ); ?>
|
| 358 |
+
<span><?php $this->_e( 'No icon for extenal links with images' ) ?></span></label>
|
| 359 |
+
</td>
|
| 360 |
+
</tr>
|
| 361 |
+
<tr>
|
| 362 |
+
<th style="width:250px;"><?php $this->_e( 'Set no-icon class' ) ?>
|
| 363 |
+
<?php echo $this->tooltip_help( 'Set this class for links, that should not have the external link icon.' ) ?></th>
|
| 364 |
+
<td><label><?php echo $this->form->text( 'no_icon_class', array( 'class' => '' ) ); ?></label>
|
| 365 |
+
<br/><label><?php echo $this->form->checkbox( 'no_icon_same_window', 1 ); ?>
|
| 366 |
+
<span><?php $this->_e( 'Always open links with no-icon class in same window or tab' ) ?></span></label>
|
| 367 |
+
<?php echo $this->tooltip_help( 'When enabled external links containing the no-icon class will always be opened in the current window or tab. No matter which target is set.' ) ?>
|
| 368 |
+
</td>
|
| 369 |
+
</tr>
|
| 370 |
+
<tr>
|
| 371 |
+
<th><?php $this->_e( 'Add to <code>class</code>-attribute' ) ?>
|
| 372 |
+
<?php echo $this->tooltip_help( 'Add one or more extra classes to the external links, seperated by a space. It is optional, else just leave field blank.' ) ?></th>
|
| 373 |
+
<td><label><?php echo $this->form->text( 'class_name' ); ?></label></td>
|
| 374 |
+
</tr>
|
| 375 |
+
</table>
|
| 376 |
+
</fieldset>
|
| 377 |
+
<?php
|
| 378 |
+
echo $this->form->submit();
|
| 379 |
+
echo $this->form->close_form();
|
| 380 |
+
}
|
| 381 |
+
|
| 382 |
+
/**
|
| 383 |
+
* Meta Box: Extra Settings
|
| 384 |
+
*/
|
| 385 |
+
public function call_box_extra_settings() {
|
| 386 |
+
echo $this->form->set_current_option( 'extra' )->open_form();
|
| 387 |
+
?>
|
| 388 |
+
<fieldset class="options">
|
| 389 |
+
<table class="form-table">
|
| 390 |
+
<tr>
|
| 391 |
+
<th style="width:250px;"><?php $this->_e( 'Solving problems' ) ?>
|
| 392 |
+
<?php echo $this->tooltip_help( 'Some options to try when a problem occurs. These options can also cause other problems, so be carefull.' ) ?></th>
|
| 393 |
+
<td><label><?php echo $this->form->checkbox( 'fix_js', 1 ); ?>
|
| 394 |
+
<span><?php $this->_e( 'Replacing <code></a></code> with <code><\/a></code> in JavaScript code.' ) ?></span></label>
|
| 395 |
+
<?php echo $this->tooltip_help( 'By replacing </a> with <\/a> in JavaScript code these tags will not be processed by the plugin.' ) ?>
|
| 396 |
+
</td>
|
| 397 |
+
</tr>
|
| 398 |
+
<tr>
|
| 399 |
+
<th style="width:250px;"><?php $this->_e( 'Use phpQuery library' ) ?>
|
| 400 |
+
<?php echo $this->tooltip_help( 'Using phpQuery library for manipulating links. This option is experimental.' ) ?></th>
|
| 401 |
+
<td><label><?php echo $this->form->checkbox( 'phpquery', 1 ); ?>
|
| 402 |
+
<span><?php $this->_e( 'Use phpQuery for parsing document.' ) ?></span>
|
| 403 |
+
<span class="description">(Test it first!)</span></label>
|
| 404 |
+
</td>
|
| 405 |
+
</tr>
|
| 406 |
+
<tr class="filter_excl_sel" <?php echo ( $this->form->value( 'phpquery' ) ) ? '' : 'style="display:none;"'; ?>>
|
| 407 |
+
<th><?php $this->_e( 'Do NOT apply settings on...' ) ?>
|
| 408 |
+
<?php echo $this->tooltip_help( 'The external links of these selection will be excluded for the settings of this plugin. Define the selection by using CSS selectors.' ) ?></th>
|
| 409 |
+
<td><label><?php echo $this->form->textarea( 'filter_excl_sel' ); ?>
|
| 410 |
+
<span class="description"><?php _e( 'Define selection by using CSS selectors, f.e.: <code>.excl-ext-link, .entry-title, #comments-title</code> (look <a href="http://code.google.com/p/phpquery/wiki/Selectors" target="_blank">here</a> for available selectors).' ) ?></span></label>
|
| 411 |
+
</td>
|
| 412 |
+
</tr>
|
| 413 |
+
</table>
|
| 414 |
+
</fieldset>
|
| 415 |
+
<?php
|
| 416 |
+
echo $this->form->submit();
|
| 417 |
+
echo $this->form->close_form();
|
| 418 |
+
}
|
| 419 |
+
|
| 420 |
+
/**
|
| 421 |
+
* Meta Box: Extra Settings
|
| 422 |
+
*/
|
| 423 |
+
public function call_box_admin_settings() {
|
| 424 |
+
echo $this->form->set_current_option( 'screen' )->open_form();
|
| 425 |
+
?>
|
| 426 |
+
<fieldset class="options">
|
| 427 |
+
<table class="form-table">
|
| 428 |
+
<tr>
|
| 429 |
+
<th><?php $this->_e('Admin menu position') ?>
|
| 430 |
+
<?php echo $this->tooltip_help( 'Change the menu position of this plugin in "Screen Options".' ) ?></th>
|
| 431 |
+
<td><label>
|
| 432 |
+
<?php
|
| 433 |
+
echo $this->form->select( 'menu_position', array(
|
| 434 |
+
'admin.php' => 'Main menu',
|
| 435 |
+
'index.php' => $this->__( 'Subitem of Dashboard' ),
|
| 436 |
+
'edit.php' => $this->__( 'Subitem of Posts' ),
|
| 437 |
+
'upload.php' => $this->__( 'Subitem of Media' ),
|
| 438 |
+
'link-manager.php' => $this->__( 'Subitem of Links' ),
|
| 439 |
+
'edit.php?post_type=page' => $this->__( 'Subitem of Pages' ),
|
| 440 |
+
'edit-comments.php' => $this->__( 'Subitem of Comments' ),
|
| 441 |
+
'themes.php' => $this->__( 'Subitem of Appearance' ),
|
| 442 |
+
'plugins.php' => $this->__( 'Subitem of Plugins' ),
|
| 443 |
+
'users.php' => $this->__( 'Subitem of Users' ),
|
| 444 |
+
'tools.php' => $this->__( 'Subitem of Tools' ),
|
| 445 |
+
'options-general.php' => $this->__( 'Subitem of Settings' ),
|
| 446 |
+
));
|
| 447 |
+
?>
|
| 448 |
+
</label></td>
|
| 449 |
+
</tr>
|
| 450 |
+
</table>
|
| 451 |
+
</fieldset>
|
| 452 |
+
<?php
|
| 453 |
+
echo $this->form->submit();
|
| 454 |
+
echo $this->form->close_form();
|
| 455 |
+
}
|
| 456 |
+
|
| 457 |
+
/**
|
| 458 |
+
* Meta Box: About...
|
| 459 |
+
*/
|
| 460 |
+
public function call_box_about() {
|
| 461 |
+
?>
|
| 462 |
+
<h4><img src="<?php echo plugins_url( 'images/icon-wp-external-links-16.png', WP_EXTERNAL_LINKS_FILE ) ?>" width="16" height="16" /> <?php $this->_e( 'WP External Links' ) ?></h4>
|
| 463 |
+
<div>
|
| 464 |
+
<p><?php printf( $this->__( 'Current version: <strong>%1$s</strong>' ), WP_EXTERNAL_LINKS_VERSION ) ?></p>
|
| 465 |
+
<p><?php $this->_e( 'Manage external links on your site: open in new window/tab, set link icon, add "external", add "nofollow" and more.' ) ?></p>
|
| 466 |
+
<p><a href="http://www.freelancephp.net/contact/" target="_blank"><?php $this->_e( 'Questions or suggestions?' ) ?></a></p>
|
| 467 |
+
<p><?php $this->_e( 'If you like this plugin please send your rating at WordPress.org.' ) ?></p>
|
| 468 |
+
<p><?php _e( 'More info' ) ?>: <a href="http://wordpress.org/extend/plugins/wp-external-links/" target="_blank">WordPress.org</a> | <a href="http://www.freelancephp.net/wp-external-links-plugin/" target="_blank">FreelancePHP.net</a></p>
|
| 469 |
+
</div>
|
| 470 |
+
<?php
|
| 471 |
+
}
|
| 472 |
+
|
| 473 |
+
/**
|
| 474 |
+
* Meta Box: Other Plugins
|
| 475 |
+
*/
|
| 476 |
+
public function call_box_other_plugins() {
|
| 477 |
+
?>
|
| 478 |
+
<h4><img src="<?php echo plugins_url( 'images/icon-email-encoder-bundle-16.png', WP_EXTERNAL_LINKS_FILE ); ?>" width="16" height="16" /> Email Encoder Bundle</h4>
|
| 479 |
+
<div>
|
| 480 |
+
<?php if ( is_plugin_active( 'email-encoder-bundle/email-encoder-bundle.php' ) ): ?>
|
| 481 |
+
<p><?php $this->_e( 'This plugin is already activated.' ) ?> <a href="<?php echo get_bloginfo( 'url' ) ?>/wp-admin/options-general.php?page=email-encoder-bundle/email-encoder-bundle.php"><?php $this->_e( 'Settings' ) ?></a></p>
|
| 482 |
+
<?php elseif( file_exists( WP_PLUGIN_DIR . '/email-encoder-bundle/email-encoder-bundle.php' ) ): ?>
|
| 483 |
+
<p><a href="<?php echo get_bloginfo( 'url' ) ?>/wp-admin/plugins.php?plugin_status=inactive"><?php $this->_e( 'Activate this plugin.' ) ?></a></p>
|
| 484 |
+
<?php else: ?>
|
| 485 |
+
<p><a href="<?php echo get_bloginfo( 'url' ) ?>/wp-admin/plugin-install.php?tab=search&type=term&s=Email+Encoder+Bundle+freelancephp&plugin-search-input=Search+Plugins"><?php $this->_e( 'Get this plugin now' ) ?></a></p>
|
| 486 |
+
<?php endif; ?>
|
| 487 |
+
|
| 488 |
+
<p><?php $this->_e( 'Protect email addresses on your site from spambots and being used for spamming by using one of the encoding methods.' ) ?></p>
|
| 489 |
+
<p><?php _e( 'More info' ) ?>: <a href="http://wordpress.org/extend/plugins/email-encoder-bundle/" target="_blank">WordPress.org</a> | <a href="http://www.freelancephp.net/email-encoder-php-class-wp-plugin/" target="_blank">FreelancePHP.net</a></p>
|
| 490 |
+
</div>
|
| 491 |
+
|
| 492 |
+
<?php echo $this->hr(); ?>
|
| 493 |
+
|
| 494 |
+
<h4><img src="<?php echo plugins_url( 'images/icon-wp-mailto-links-16.png', WP_EXTERNAL_LINKS_FILE ); ?>" width="16" height="16" /> WP Mailto Links</h4>
|
| 495 |
+
<div>
|
| 496 |
+
<?php if ( is_plugin_active( 'wp-mailto-links/wp-mailto-links.php' ) ): ?>
|
| 497 |
+
<p><?php $this->_e( 'This plugin is already activated.' ) ?> <a href="<?php echo get_bloginfo( 'url' ) ?>/wp-admin/options-general.php?page=wp-mailto-links/wp-mailto-links.php"><?php $this->_e( 'Settings' ) ?></a></p>
|
| 498 |
+
<?php elseif( file_exists( WP_PLUGIN_DIR . '/wp-mailto-links/wp-mailto-links.php' ) ): ?>
|
| 499 |
+
<p><a href="<?php echo get_bloginfo( 'url' ) ?>/wp-admin/plugins.php?plugin_status=inactive"><?php $this->_e( 'Activate this plugin.' ) ?></a></p>
|
| 500 |
+
<?php else: ?>
|
| 501 |
+
<p><a href="<?php echo get_bloginfo( 'url' ) ?>/wp-admin/plugin-install.php?tab=search&type=term&s=WP+Mailto+Links+freelancephp&plugin-search-input=Search+Plugins"><?php $this->_e( 'Get this plugin now' ) ?></a></p>
|
| 502 |
+
<?php endif; ?>
|
| 503 |
+
|
| 504 |
+
<p><?php $this->_e( 'Manage mailto links on your site and protect emails from spambots, set mail icon and more.' ) ?></p>
|
| 505 |
+
<p><?php _e( 'More info' ) ?>: <a href="http://wordpress.org/extend/plugins/wp-mailto-links/" target="_blank">WordPress.org</a> | <a href="http://www.freelancephp.net/wp-mailto-links-plugin/" target="_blank">FreelancePHP.net</a></p>
|
| 506 |
+
</div>
|
| 507 |
+
<?php
|
| 508 |
+
}
|
| 509 |
+
|
| 510 |
+
/**
|
| 511 |
+
* Activation callback
|
| 512 |
+
*/
|
| 513 |
+
public function check_version_update() {
|
| 514 |
+
// check for version
|
| 515 |
+
$meta = get_option( 'wp_external_links-meta' );
|
| 516 |
+
if ( $meta[ 'version' ] == WP_EXTERNAL_LINKS_VERSION )
|
| 517 |
+
return;
|
| 518 |
+
|
| 519 |
+
// set new version
|
| 520 |
+
$meta[ 'version' ] = WP_EXTERNAL_LINKS_VERSION;
|
| 521 |
+
update_option( 'wp_external_links-meta', $meta );
|
| 522 |
+
|
| 523 |
+
// check for upgrading saved options to v1.00
|
| 524 |
+
$old_options = get_option( 'WP_External_Links_options' );
|
| 525 |
+
|
| 526 |
+
if ( ! empty( $old_options ) ) {
|
| 527 |
+
$new_options = $this->save_options;
|
| 528 |
+
|
| 529 |
+
$new_options[ 'main' ][ 'target' ] = $old_options[ 'target' ];
|
| 530 |
+
$new_options[ 'main' ][ 'filter_page' ] = $old_options[ 'filter_whole_page' ];
|
| 531 |
+
$new_options[ 'main' ][ 'filter_posts' ] = $old_options[ 'filter_posts' ];
|
| 532 |
+
$new_options[ 'main' ][ 'filter_comments' ] = $old_options[ 'filter_comments' ];
|
| 533 |
+
$new_options[ 'main' ][ 'filter_widgets' ] = $old_options[ 'filter_widgets' ];
|
| 534 |
+
$new_options[ 'seo' ][ 'external' ] = $old_options[ 'external' ];
|
| 535 |
+
$new_options[ 'seo' ][ 'nofollow' ] = $old_options[ 'nofollow' ];
|
| 536 |
+
$new_options[ 'seo' ][ 'use_js' ] = $old_options[ 'use_js' ];
|
| 537 |
+
$new_options[ 'style' ][ 'class_name' ] = $old_options[ 'class_name' ];
|
| 538 |
+
$new_options[ 'style' ][ 'icon' ] = $old_options[ 'icon' ];
|
| 539 |
+
$new_options[ 'style' ][ 'no_icon_class' ] = $old_options[ 'no_icon_class' ];
|
| 540 |
+
$new_options[ 'style' ][ 'no_icon_same_window' ] = $old_options[ 'no_icon_same_window' ];
|
| 541 |
+
|
| 542 |
+
// save new format option values
|
| 543 |
+
update_option( 'wp_external_links-main', $new_options[ 'main' ] );
|
| 544 |
+
update_option( 'wp_external_links-seo', $new_options[ 'seo' ] );
|
| 545 |
+
update_option( 'wp_external_links-style', $new_options[ 'style' ] );
|
| 546 |
+
|
| 547 |
+
// delete old format option values
|
| 548 |
+
delete_option( 'WP_External_Links_options' );
|
| 549 |
+
}
|
| 550 |
+
|
| 551 |
+
// upgrade to v1.20
|
| 552 |
+
$upgrade_main = get_option( 'wp_external_links-main' );
|
| 553 |
+
|
| 554 |
+
if ( ! isset( $upgrade_main[ 'ignore' ] ) ) {
|
| 555 |
+
$upgrade_main[ 'ignore' ] = $this->save_options[ 'main' ][ 'ignore' ];
|
| 556 |
+
update_option( 'wp_external_links-main', $upgrade_main );
|
| 557 |
+
}
|
| 558 |
+
|
| 559 |
+
// upgrade to v1.30
|
| 560 |
+
if ( WP_EXTERNAL_LINKS_VERSION == '1.30' ) {
|
| 561 |
+
$new_options = $this->save_options;
|
| 562 |
+
$general = get_option( 'wp_external_links-general' );
|
| 563 |
+
$style = get_option( 'wp_external_links-style' );
|
| 564 |
+
|
| 565 |
+
if ( isset( $general[ 'target' ] ) ) $new_options[ 'main' ][ 'target' ] = $general[ 'target' ];
|
| 566 |
+
$new_options[ 'main' ][ 'filter_page' ] = ( isset( $general[ 'filter_page' ] ) ) ? $general[ 'filter_page' ] : 0;
|
| 567 |
+
$new_options[ 'main' ][ 'filter_posts' ] = ( isset( $general[ 'filter_posts' ] ) ) ? $general[ 'filter_posts' ] : 0;
|
| 568 |
+
$new_options[ 'main' ][ 'filter_comments' ] = ( isset( $general[ 'filter_comments' ] ) ) ? $general[ 'filter_comments' ] : 0;
|
| 569 |
+
$new_options[ 'main' ][ 'filter_widgets' ] = ( isset( $general[ 'filter_widgets' ] ) ) ? $general[ 'filter_widgets' ] : 0;
|
| 570 |
+
if ( isset( $general[ 'ignore' ] ) ) $new_options[ 'main' ][ 'ignore' ] = $general[ 'ignore' ];
|
| 571 |
+
|
| 572 |
+
$new_options[ 'seo' ][ 'external' ] = ( isset( $general[ 'external' ] ) ) ? $general[ 'external' ] : 0;
|
| 573 |
+
$new_options[ 'seo' ][ 'nofollow' ] = ( isset( $general[ 'nofollow' ] ) ) ? $general[ 'nofollow' ] : 0;
|
| 574 |
+
$new_options[ 'seo' ][ 'use_js' ] = ( isset( $general[ 'use_js' ] ) ) ? $general[ 'use_js' ] : 0;
|
| 575 |
+
if ( isset( $general[ 'title' ] ) ) $new_options[ 'seo' ][ 'title' ] = $general[ 'title' ];
|
| 576 |
+
|
| 577 |
+
if ( isset( $general[ 'class_name' ] ) ) $new_options[ 'style' ][ 'class_name' ] = $general[ 'class_name' ];
|
| 578 |
+
|
| 579 |
+
if ( isset( $style[ 'icon' ] ) ) $new_options[ 'style' ][ 'icon' ] = $style[ 'icon' ];
|
| 580 |
+
if ( isset( $style[ 'no_icon_class' ] ) ) $new_options[ 'style' ][ 'no_icon_class' ] = $style[ 'no_icon_class' ];
|
| 581 |
+
$new_options[ 'style' ][ 'no_icon_same_window' ] = ( isset( $style[ 'no_icon_same_window' ] ) ) ? $style[ 'no_icon_same_window' ] : 0;
|
| 582 |
+
|
| 583 |
+
$new_options[ 'extra' ][ 'fix_js' ] = ( isset( $general[ 'fix_js' ] ) ) ? $general[ 'fix_js' ] : 0;
|
| 584 |
+
$new_options[ 'extra' ][ 'phpquery' ] = ( isset( $general[ 'phpquery' ] ) ) ? $general[ 'phpquery' ] : 0;
|
| 585 |
+
if ( isset( $general[ 'filter_excl_sel' ] ) ) $new_options[ 'extra' ][ 'filter_excl_sel' ] = $general[ 'filter_excl_sel' ];
|
| 586 |
+
|
| 587 |
+
// save new format option values
|
| 588 |
+
update_option( 'wp_external_links-main', $new_options[ 'main' ] );
|
| 589 |
+
update_option( 'wp_external_links-seo', $new_options[ 'seo' ] );
|
| 590 |
+
update_option( 'wp_external_links-style', $new_options[ 'style' ] );
|
| 591 |
+
update_option( 'wp_external_links-extra', $new_options[ 'extra' ] );
|
| 592 |
+
|
| 593 |
+
// delete old format
|
| 594 |
+
delete_option( 'wp_external_links-general' );
|
| 595 |
+
}
|
| 596 |
+
}
|
| 597 |
+
|
| 598 |
+
/**
|
| 599 |
+
* Method for test purpuses
|
| 600 |
+
*/
|
| 601 |
+
public function __options($values = null) {
|
| 602 |
+
if (class_exists('Test_WP_Mailto_Links') && constant('WP_DEBUG') === true) {
|
| 603 |
+
if ($values !== null) {
|
| 604 |
+
$this->set_options($values);
|
| 605 |
+
}
|
| 606 |
+
|
| 607 |
+
return $this->options;
|
| 608 |
+
}
|
| 609 |
+
}
|
| 610 |
+
|
| 611 |
+
/**
|
| 612 |
+
* Uninstall callback
|
| 613 |
+
*/
|
| 614 |
+
static public function call_uninstall() {
|
| 615 |
+
self::$staticForm->delete_options();
|
| 616 |
+
}
|
| 617 |
+
|
| 618 |
+
/**
|
| 619 |
+
* Set tooltip help
|
| 620 |
+
* @param string $text
|
| 621 |
+
* @return string
|
| 622 |
+
*/
|
| 623 |
+
public function tooltip_help( $text ) {
|
| 624 |
+
$text = $this->__( $text );
|
| 625 |
+
$text = htmlentities( $text );
|
| 626 |
+
|
| 627 |
+
$html = '<a href="#" class="tooltip-help" title="'. $text .'"><sup>(?)</sup></a>';
|
| 628 |
+
return $html;
|
| 629 |
+
}
|
| 630 |
+
|
| 631 |
+
/**
|
| 632 |
+
* Get html seperator
|
| 633 |
+
* @return string
|
| 634 |
+
*/
|
| 635 |
+
protected function hr() {
|
| 636 |
+
return '<hr style="border:1px solid #FFF; border-top:1px solid #EEE;" />';
|
| 637 |
+
}
|
| 638 |
+
|
| 639 |
+
|
| 640 |
+
/**
|
| 641 |
+
* Check if widget_content filter is available (Widget Logic Plugin)
|
| 642 |
+
* @return boolean
|
| 643 |
+
* @static
|
| 644 |
+
*/
|
| 645 |
+
public static function check_widget_content_filter() {
|
| 646 |
+
// set widget_content filter of Widget Logic plugin
|
| 647 |
+
$widget_logic_opts = get_option( 'widget_logic' );
|
| 648 |
+
|
| 649 |
+
if ( function_exists( 'widget_logic_expand_control' ) AND is_array( $widget_logic_opts ) AND key_exists( 'widget_logic-options-filter', $widget_logic_opts ) )
|
| 650 |
+
return ( $widget_logic_opts[ 'widget_logic-options-filter' ] == 'checked' );
|
| 651 |
+
|
| 652 |
+
return FALSE;
|
| 653 |
+
}
|
| 654 |
+
|
| 655 |
+
} // End Admin_External_Links Class
|
| 656 |
+
|
| 657 |
+
endif;
|
| 658 |
+
|
| 659 |
/* ommit PHP closing tag, to prevent unwanted whitespace at the end of the parts generated by the included files */
|
includes/class-wp-external-links.php
CHANGED
|
@@ -1,594 +1,604 @@
|
|
| 1 |
-
<?php defined( 'ABSPATH' ) OR die( 'No direct access.' );
|
| 2 |
-
if ( ! class_exists( 'WP_External_Links' ) ):
|
| 3 |
-
|
| 4 |
-
/**
|
| 5 |
-
* Class WP_External_Links
|
| 6 |
-
* @package WordPress
|
| 7 |
-
* @since
|
| 8 |
-
* @category WordPress Plugins
|
| 9 |
-
*/
|
| 10 |
-
final class WP_External_Links {
|
| 11 |
-
|
| 12 |
-
/**
|
| 13 |
-
* Admin object
|
| 14 |
-
* @var Admin_External_Links
|
| 15 |
-
*/
|
| 16 |
-
public $admin = NULL;
|
| 17 |
-
|
| 18 |
-
/**
|
| 19 |
-
* Array of ignored links
|
| 20 |
-
* @var type
|
| 21 |
-
*/
|
| 22 |
-
private $ignored = array();
|
| 23 |
-
|
| 24 |
-
|
| 25 |
-
/**
|
| 26 |
-
* Constructor
|
| 27 |
-
*/
|
| 28 |
-
public function __construct() {
|
| 29 |
-
// set admin object
|
| 30 |
-
$this->admin = new Admin_External_Links();
|
| 31 |
-
|
| 32 |
-
// add actions
|
| 33 |
-
add_action( 'wp', array( $this, 'call_wp' ) );
|
| 34 |
-
}
|
| 35 |
-
|
| 36 |
-
/**
|
| 37 |
-
* Quick helper method for getting saved option values
|
| 38 |
-
* @param string $key
|
| 39 |
-
* @return mixed
|
| 40 |
-
*/
|
| 41 |
-
public function get_opt( $key ) {
|
| 42 |
-
$lookup = $this->admin->save_options;
|
| 43 |
-
|
| 44 |
-
foreach ( $lookup as $option_name => $values ) {
|
| 45 |
-
$value = $this->admin->form->value( $key, '___NONE___', $option_name );
|
| 46 |
-
|
| 47 |
-
if ($value !== '___NONE___')
|
| 48 |
-
return $value;
|
| 49 |
-
}
|
| 50 |
-
|
| 51 |
-
throw new Exception('Option with key "' . $key . '" does not exist.');
|
| 52 |
-
}
|
| 53 |
-
|
| 54 |
-
/**
|
| 55 |
-
* wp callback
|
| 56 |
-
*/
|
| 57 |
-
public function call_wp() {
|
| 58 |
-
if ( ! is_admin() && ! is_feed() ) {
|
| 59 |
-
// add wp_head for setting js vars and css style
|
| 60 |
-
add_action( 'wp_head', array( $this, 'call_wp_head' ) );
|
| 61 |
-
|
| 62 |
-
// set js file
|
| 63 |
-
if ( $this->get_opt( 'use_js' ) )
|
| 64 |
-
wp_enqueue_script( 'wp-external-links', plugins_url( 'js/wp-external-links.js', WP_EXTERNAL_LINKS_FILE ), array(
|
| 65 |
-
|
| 66 |
-
// filters
|
| 67 |
-
if ( $this->get_opt( 'filter_page' ) ) {
|
| 68 |
-
// filter body
|
| 69 |
-
ob_start( array( $this, 'call_filter_content' ) );
|
| 70 |
-
|
| 71 |
-
// set ob flush
|
| 72 |
-
add_action('wp_footer', array($this, 'callback_flush_buffer'), 10000);
|
| 73 |
-
|
| 74 |
-
} else {
|
| 75 |
-
// set filter priority
|
| 76 |
-
$priority = 1000000000;
|
| 77 |
-
|
| 78 |
-
// content
|
| 79 |
-
if ( $this->get_opt( 'filter_posts' ) ) {
|
| 80 |
-
add_filter( 'the_title', array( $this, 'call_filter_content' ), $priority );
|
| 81 |
-
add_filter( 'the_content', array( $this, 'call_filter_content' ), $priority );
|
| 82 |
-
add_filter( 'get_the_excerpt', array( $this, 'call_filter_content' ), $priority );
|
| 83 |
-
// redundant:
|
| 84 |
-
//add_filter( 'the_excerpt', array( $this, 'call_filter_content' ), $priority );
|
| 85 |
-
}
|
| 86 |
-
|
| 87 |
-
// comments
|
| 88 |
-
if ( $this->get_opt( 'filter_comments' ) ) {
|
| 89 |
-
add_filter( 'get_comment_text', array( $this, 'call_filter_content' ), $priority );
|
| 90 |
-
// redundant:
|
| 91 |
-
//add_filter( 'comment_text', array( $this, 'call_filter_content' ), $priority );
|
| 92 |
-
|
| 93 |
-
add_filter( 'comment_excerpt', array( $this, 'call_filter_content' ), $priority );
|
| 94 |
-
// redundant:
|
| 95 |
-
//add_filter( 'get_comment_excerpt', array( $this, 'call_filter_content' ), $priority );
|
| 96 |
-
|
| 97 |
-
add_filter( 'comment_url', array( $this, 'call_filter_content' ), $priority );
|
| 98 |
-
add_filter( 'get_comment_author_url', array( $this, 'call_filter_content' ), $priority );
|
| 99 |
-
add_filter( 'get_comment_author_link', array( $this, 'call_filter_content' ), $priority );
|
| 100 |
-
add_filter( 'get_comment_author_url_link', array( $this, 'call_filter_content' ), $priority );
|
| 101 |
-
}
|
| 102 |
-
|
| 103 |
-
// widgets
|
| 104 |
-
if ( $this->get_opt( 'filter_widgets' ) ) {
|
| 105 |
-
if ( $this->admin->check_widget_content_filter() ) {
|
| 106 |
-
// only if Widget Logic plugin is installed and 'widget_content' option is activated
|
| 107 |
-
add_filter( 'widget_content', array( $this, 'call_filter_content' ), $priority );
|
| 108 |
-
} else {
|
| 109 |
-
// filter text widgets
|
| 110 |
-
add_filter( 'widget_title', array( $this, 'call_filter_content' ), $priority );
|
| 111 |
-
add_filter( 'widget_text', array( $this, 'call_filter_content' ), $priority );
|
| 112 |
-
}
|
| 113 |
-
}
|
| 114 |
-
}
|
| 115 |
-
}
|
| 116 |
-
|
| 117 |
-
// hook
|
| 118 |
-
do_action('wpel_ready', array($this, 'call_filter_content'), $this);
|
| 119 |
-
}
|
| 120 |
-
|
| 121 |
-
/**
|
| 122 |
-
* End output buffer
|
| 123 |
-
*/
|
| 124 |
-
public function callback_flush_buffer() {
|
| 125 |
-
ob_end_flush();
|
| 126 |
-
}
|
| 127 |
-
|
| 128 |
-
/**
|
| 129 |
-
* wp_head callback
|
| 130 |
-
*/
|
| 131 |
-
public function call_wp_head() {
|
| 132 |
-
// set ignored
|
| 133 |
-
$ignored = $this->get_opt( 'ignore' );
|
| 134 |
-
$ignored = trim( $ignored );
|
| 135 |
-
$ignored = explode( "\n", $ignored );
|
| 136 |
-
$ignored = array_map( 'trim', $ignored );
|
| 137 |
-
$ignored = array_map( 'strtolower', $ignored );
|
| 138 |
-
$this->ignored = $ignored;
|
| 139 |
-
|
| 140 |
-
$icon = $this->get_opt('icon');
|
| 141 |
-
|
| 142 |
-
if ($icon) {
|
| 143 |
-
$padding = ($icon < 20) ? 15 : 12;
|
| 144 |
-
?>
|
| 145 |
-
<style type="text/css" media="screen">
|
| 146 |
-
/* WP External Links Plugin */
|
| 147 |
-
.ext-icon-<?php echo $icon ?> { background:url(<?php echo plugins_url('/images/ext-icons/ext-icon-' . $icon . '.png', WP_EXTERNAL_LINKS_FILE) ?>) no-repeat 100% 50%; padding-right:<?php echo $padding ?>px; }';
|
| 148 |
-
</style>
|
| 149 |
-
<?php
|
| 150 |
-
}
|
| 151 |
-
|
| 152 |
-
if ( $this->get_opt( 'use_js' ) AND $this->get_opt( 'target' ) != '_none' ):
|
| 153 |
-
// set exclude class
|
| 154 |
-
$excludeClass = ( $this->get_opt( 'no_icon_same_window' ) AND $this->get_opt( 'no_icon_class' ) )
|
| 155 |
-
? $this->get_opt( 'no_icon_class' )
|
| 156 |
-
: '';
|
| 157 |
-
?>
|
| 158 |
-
<script type="text/javascript">/* <![CDATA[ */
|
| 159 |
-
/* WP External Links Plugin */
|
| 160 |
-
var wpExtLinks = { baseUrl: '<?php echo get_bloginfo( 'wpurl' ) ?>', target: '<?php echo $this->get_opt( 'target' ) ?>', excludeClass: '<?php echo $excludeClass ?>' };
|
| 161 |
-
/* ]]> */</script>
|
| 162 |
-
<?php
|
| 163 |
-
endif;
|
| 164 |
-
}
|
| 165 |
-
|
| 166 |
-
/**
|
| 167 |
-
* Filter content
|
| 168 |
-
* @param string $content
|
| 169 |
-
* @return string
|
| 170 |
-
*/
|
| 171 |
-
public function call_filter_content( $content ) {
|
| 172 |
-
if ( $this->get_opt( 'fix_js' ) ) {
|
| 173 |
-
// fix js problem by replacing </a> by <\/a>
|
| 174 |
-
$content = preg_replace_callback( '/<script([^>]*)>(.*?)<\/script[^>]*>/is', array( $this, 'call_fix_js' ), $content );
|
| 175 |
-
}
|
| 176 |
-
|
| 177 |
-
if ( $this->get_opt( 'phpquery' ) ) {
|
| 178 |
-
// Include phpQuery
|
| 179 |
-
if ( ! class_exists( 'phpQuery' ) ) {
|
| 180 |
-
require_once( 'phpQuery.php' );
|
| 181 |
-
}
|
| 182 |
-
|
| 183 |
-
return $this->filter_phpquery( $content );
|
| 184 |
-
} else {
|
| 185 |
-
return $this->filter( $content );
|
| 186 |
-
}
|
| 187 |
-
}
|
| 188 |
-
|
| 189 |
-
/**
|
| 190 |
-
* Fix </a> in JavaScript blocks (callback for regexp)
|
| 191 |
-
* @param array $matches Result of a preg call in filter_content()
|
| 192 |
-
* @return string Clean code
|
| 193 |
-
*/
|
| 194 |
-
public function call_fix_js( $matches ) {
|
| 195 |
-
return str_replace( '</a>', '<\/a>', $matches[ 0 ] );
|
| 196 |
-
}
|
| 197 |
-
|
| 198 |
-
/**
|
| 199 |
-
* Check if link is external
|
| 200 |
-
* @param string $href
|
| 201 |
-
* @param string $rel
|
| 202 |
-
* @return boolean
|
| 203 |
-
*/
|
| 204 |
-
private function is_external( $href, $rel ) {
|
| 205 |
-
return ( isset( $href ) AND ( strpos( $
|
| 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 |
-
* @example parse_attrs( 'src="example.jpg" alt="example"' )
|
| 255 |
-
* @example parse_attrs( '<
|
| 256 |
-
* @example parse_attrs( '<a href="example"
|
| 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 |
-
if ( $
|
| 372 |
-
$this->add_attr_value( $attrs, '
|
| 373 |
-
|
| 374 |
-
// set
|
| 375 |
-
|
| 376 |
-
|
| 377 |
-
|
| 378 |
-
|
| 379 |
-
|
| 380 |
-
|
| 381 |
-
|
| 382 |
-
|
| 383 |
-
|
| 384 |
-
|
| 385 |
-
|
| 386 |
-
OR strpos( $attrs[ 'class' ], $this->get_opt( 'no_icon_class' ) ) === FALSE )
|
| 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 |
-
// set
|
| 468 |
-
|
| 469 |
-
|
| 470 |
-
|
| 471 |
-
|
| 472 |
-
|
| 473 |
-
|
| 474 |
-
|
| 475 |
-
|
| 476 |
-
|
| 477 |
-
|
| 478 |
-
|
| 479 |
-
|
| 480 |
-
$
|
| 481 |
-
|
| 482 |
-
// set
|
| 483 |
-
|
| 484 |
-
|
| 485 |
-
|
| 486 |
-
$a
|
| 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 |
-
$a->
|
| 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 |
/* ommit PHP closing tag, to prevent unwanted whitespace at the end of the parts generated by the included files */
|
| 1 |
+
<?php defined( 'ABSPATH' ) OR die( 'No direct access.' );
|
| 2 |
+
if ( ! class_exists( 'WP_External_Links' ) ):
|
| 3 |
+
|
| 4 |
+
/**
|
| 5 |
+
* Class WP_External_Links
|
| 6 |
+
* @package WordPress
|
| 7 |
+
* @since
|
| 8 |
+
* @category WordPress Plugins
|
| 9 |
+
*/
|
| 10 |
+
final class WP_External_Links {
|
| 11 |
+
|
| 12 |
+
/**
|
| 13 |
+
* Admin object
|
| 14 |
+
* @var Admin_External_Links
|
| 15 |
+
*/
|
| 16 |
+
public $admin = NULL;
|
| 17 |
+
|
| 18 |
+
/**
|
| 19 |
+
* Array of ignored links
|
| 20 |
+
* @var type
|
| 21 |
+
*/
|
| 22 |
+
private $ignored = array();
|
| 23 |
+
|
| 24 |
+
|
| 25 |
+
/**
|
| 26 |
+
* Constructor
|
| 27 |
+
*/
|
| 28 |
+
public function __construct() {
|
| 29 |
+
// set admin object
|
| 30 |
+
$this->admin = new Admin_External_Links();
|
| 31 |
+
|
| 32 |
+
// add actions
|
| 33 |
+
add_action( 'wp', array( $this, 'call_wp' ) );
|
| 34 |
+
}
|
| 35 |
+
|
| 36 |
+
/**
|
| 37 |
+
* Quick helper method for getting saved option values
|
| 38 |
+
* @param string $key
|
| 39 |
+
* @return mixed
|
| 40 |
+
*/
|
| 41 |
+
public function get_opt( $key ) {
|
| 42 |
+
$lookup = $this->admin->save_options;
|
| 43 |
+
|
| 44 |
+
foreach ( $lookup as $option_name => $values ) {
|
| 45 |
+
$value = $this->admin->form->value( $key, '___NONE___', $option_name );
|
| 46 |
+
|
| 47 |
+
if ($value !== '___NONE___')
|
| 48 |
+
return $value;
|
| 49 |
+
}
|
| 50 |
+
|
| 51 |
+
throw new Exception('Option with key "' . $key . '" does not exist.');
|
| 52 |
+
}
|
| 53 |
+
|
| 54 |
+
/**
|
| 55 |
+
* wp callback
|
| 56 |
+
*/
|
| 57 |
+
public function call_wp() {
|
| 58 |
+
if ( ! is_admin() && ! is_feed() ) {
|
| 59 |
+
// add wp_head for setting js vars and css style
|
| 60 |
+
add_action( 'wp_head', array( $this, 'call_wp_head' ) );
|
| 61 |
+
|
| 62 |
+
// set js file
|
| 63 |
+
if ( $this->get_opt( 'use_js' ) )
|
| 64 |
+
wp_enqueue_script( 'wp-external-links', plugins_url( 'js/wp-external-links.js', WP_EXTERNAL_LINKS_FILE ), array(), WP_EXTERNAL_LINKS_VERSION, (bool) $this->get_opt( 'load_in_footer' ) );
|
| 65 |
+
|
| 66 |
+
// filters
|
| 67 |
+
if ( $this->get_opt( 'filter_page' ) ) {
|
| 68 |
+
// filter body
|
| 69 |
+
ob_start( array( $this, 'call_filter_content' ) );
|
| 70 |
+
|
| 71 |
+
// set ob flush
|
| 72 |
+
add_action('wp_footer', array($this, 'callback_flush_buffer'), 10000);
|
| 73 |
+
|
| 74 |
+
} else {
|
| 75 |
+
// set filter priority
|
| 76 |
+
$priority = 1000000000;
|
| 77 |
+
|
| 78 |
+
// content
|
| 79 |
+
if ( $this->get_opt( 'filter_posts' ) ) {
|
| 80 |
+
add_filter( 'the_title', array( $this, 'call_filter_content' ), $priority );
|
| 81 |
+
add_filter( 'the_content', array( $this, 'call_filter_content' ), $priority );
|
| 82 |
+
add_filter( 'get_the_excerpt', array( $this, 'call_filter_content' ), $priority );
|
| 83 |
+
// redundant:
|
| 84 |
+
//add_filter( 'the_excerpt', array( $this, 'call_filter_content' ), $priority );
|
| 85 |
+
}
|
| 86 |
+
|
| 87 |
+
// comments
|
| 88 |
+
if ( $this->get_opt( 'filter_comments' ) ) {
|
| 89 |
+
add_filter( 'get_comment_text', array( $this, 'call_filter_content' ), $priority );
|
| 90 |
+
// redundant:
|
| 91 |
+
//add_filter( 'comment_text', array( $this, 'call_filter_content' ), $priority );
|
| 92 |
+
|
| 93 |
+
add_filter( 'comment_excerpt', array( $this, 'call_filter_content' ), $priority );
|
| 94 |
+
// redundant:
|
| 95 |
+
//add_filter( 'get_comment_excerpt', array( $this, 'call_filter_content' ), $priority );
|
| 96 |
+
|
| 97 |
+
add_filter( 'comment_url', array( $this, 'call_filter_content' ), $priority );
|
| 98 |
+
add_filter( 'get_comment_author_url', array( $this, 'call_filter_content' ), $priority );
|
| 99 |
+
add_filter( 'get_comment_author_link', array( $this, 'call_filter_content' ), $priority );
|
| 100 |
+
add_filter( 'get_comment_author_url_link', array( $this, 'call_filter_content' ), $priority );
|
| 101 |
+
}
|
| 102 |
+
|
| 103 |
+
// widgets
|
| 104 |
+
if ( $this->get_opt( 'filter_widgets' ) ) {
|
| 105 |
+
if ( $this->admin->check_widget_content_filter() ) {
|
| 106 |
+
// only if Widget Logic plugin is installed and 'widget_content' option is activated
|
| 107 |
+
add_filter( 'widget_content', array( $this, 'call_filter_content' ), $priority );
|
| 108 |
+
} else {
|
| 109 |
+
// filter text widgets
|
| 110 |
+
add_filter( 'widget_title', array( $this, 'call_filter_content' ), $priority );
|
| 111 |
+
add_filter( 'widget_text', array( $this, 'call_filter_content' ), $priority );
|
| 112 |
+
}
|
| 113 |
+
}
|
| 114 |
+
}
|
| 115 |
+
}
|
| 116 |
+
|
| 117 |
+
// hook
|
| 118 |
+
do_action('wpel_ready', array($this, 'call_filter_content'), $this);
|
| 119 |
+
}
|
| 120 |
+
|
| 121 |
+
/**
|
| 122 |
+
* End output buffer
|
| 123 |
+
*/
|
| 124 |
+
public function callback_flush_buffer() {
|
| 125 |
+
ob_end_flush();
|
| 126 |
+
}
|
| 127 |
+
|
| 128 |
+
/**
|
| 129 |
+
* wp_head callback
|
| 130 |
+
*/
|
| 131 |
+
public function call_wp_head() {
|
| 132 |
+
// set ignored
|
| 133 |
+
$ignored = $this->get_opt( 'ignore' );
|
| 134 |
+
$ignored = trim( $ignored );
|
| 135 |
+
$ignored = explode( "\n", $ignored );
|
| 136 |
+
$ignored = array_map( 'trim', $ignored );
|
| 137 |
+
$ignored = array_map( 'strtolower', $ignored );
|
| 138 |
+
$this->ignored = $ignored;
|
| 139 |
+
|
| 140 |
+
$icon = $this->get_opt('icon');
|
| 141 |
+
|
| 142 |
+
if ($icon) {
|
| 143 |
+
$padding = ($icon < 20) ? 15 : 12;
|
| 144 |
+
?>
|
| 145 |
+
<style type="text/css" media="screen">
|
| 146 |
+
/* WP External Links Plugin */
|
| 147 |
+
.ext-icon-<?php echo $icon ?> { background:url(<?php echo plugins_url('/images/ext-icons/ext-icon-' . $icon . '.png', WP_EXTERNAL_LINKS_FILE) ?>) no-repeat 100% 50%; padding-right:<?php echo $padding ?>px; }';
|
| 148 |
+
</style>
|
| 149 |
+
<?php
|
| 150 |
+
}
|
| 151 |
+
|
| 152 |
+
if ( $this->get_opt( 'use_js' ) AND $this->get_opt( 'target' ) != '_none' ):
|
| 153 |
+
// set exclude class
|
| 154 |
+
$excludeClass = ( $this->get_opt( 'no_icon_same_window' ) AND $this->get_opt( 'no_icon_class' ) )
|
| 155 |
+
? $this->get_opt( 'no_icon_class' )
|
| 156 |
+
: '';
|
| 157 |
+
?>
|
| 158 |
+
<script type="text/javascript">/* <![CDATA[ */
|
| 159 |
+
/* WP External Links Plugin */
|
| 160 |
+
var wpExtLinks = { baseUrl: '<?php echo get_bloginfo( 'wpurl' ) ?>', target: '<?php echo $this->get_opt( 'target' ) ?>', excludeClass: '<?php echo $excludeClass ?>' };
|
| 161 |
+
/* ]]> */</script>
|
| 162 |
+
<?php
|
| 163 |
+
endif;
|
| 164 |
+
}
|
| 165 |
+
|
| 166 |
+
/**
|
| 167 |
+
* Filter content
|
| 168 |
+
* @param string $content
|
| 169 |
+
* @return string
|
| 170 |
+
*/
|
| 171 |
+
public function call_filter_content( $content ) {
|
| 172 |
+
if ( $this->get_opt( 'fix_js' ) ) {
|
| 173 |
+
// fix js problem by replacing </a> by <\/a>
|
| 174 |
+
$content = preg_replace_callback( '/<script([^>]*)>(.*?)<\/script[^>]*>/is', array( $this, 'call_fix_js' ), $content );
|
| 175 |
+
}
|
| 176 |
+
|
| 177 |
+
if ( $this->get_opt( 'phpquery' ) ) {
|
| 178 |
+
// Include phpQuery
|
| 179 |
+
if ( ! class_exists( 'phpQuery' ) ) {
|
| 180 |
+
require_once( 'phpQuery.php' );
|
| 181 |
+
}
|
| 182 |
+
|
| 183 |
+
return $this->filter_phpquery( $content );
|
| 184 |
+
} else {
|
| 185 |
+
return $this->filter( $content );
|
| 186 |
+
}
|
| 187 |
+
}
|
| 188 |
+
|
| 189 |
+
/**
|
| 190 |
+
* Fix </a> in JavaScript blocks (callback for regexp)
|
| 191 |
+
* @param array $matches Result of a preg call in filter_content()
|
| 192 |
+
* @return string Clean code
|
| 193 |
+
*/
|
| 194 |
+
public function call_fix_js( $matches ) {
|
| 195 |
+
return str_replace( '</a>', '<\/a>', $matches[ 0 ] );
|
| 196 |
+
}
|
| 197 |
+
|
| 198 |
+
/**
|
| 199 |
+
* Check if link is external
|
| 200 |
+
* @param string $href
|
| 201 |
+
* @param string $rel
|
| 202 |
+
* @return boolean
|
| 203 |
+
*/
|
| 204 |
+
private function is_external( $href, $rel ) {
|
| 205 |
+
return ( isset( $href ) AND ( ( strpos( $href, strtolower( get_bloginfo( 'wpurl' ) ) ) === FALSE )
|
| 206 |
+
AND ( substr( $href, 0, 7 ) == 'http://'
|
| 207 |
+
OR substr( $href, 0, 8 ) == 'https://'
|
| 208 |
+
OR substr( $href, 0, 6 ) == 'ftp://'
|
| 209 |
+
OR substr( $href, 0, 2 ) == '//' ) ) );
|
| 210 |
+
}
|
| 211 |
+
|
| 212 |
+
/**
|
| 213 |
+
* Is an ignored link
|
| 214 |
+
* @param string $href
|
| 215 |
+
* @return boolean
|
| 216 |
+
*/
|
| 217 |
+
private function is_ignored( $href ) {
|
| 218 |
+
// check if this links should be ignored
|
| 219 |
+
for ( $x = 0, $count = count($this->ignored); $x < $count; $x++ ) {
|
| 220 |
+
if ( strrpos( $href, $this->ignored[ $x ] ) !== FALSE )
|
| 221 |
+
return TRUE;
|
| 222 |
+
}
|
| 223 |
+
|
| 224 |
+
return FALSE;
|
| 225 |
+
}
|
| 226 |
+
|
| 227 |
+
/**
|
| 228 |
+
* Filter content
|
| 229 |
+
* @param string $content
|
| 230 |
+
* @return string
|
| 231 |
+
*/
|
| 232 |
+
private function filter( $content ) {
|
| 233 |
+
// replace links
|
| 234 |
+
$content = preg_replace_callback( '/<a[^A-Za-z](.*?)>(.*?)<\/a[\s+]*>/is', array( $this, 'call_parse_link' ), $content );
|
| 235 |
+
|
| 236 |
+
// remove style when no icon classes are found
|
| 237 |
+
if ( strpos( $content, 'ext-icon-' ) === FALSE ) {
|
| 238 |
+
// remove style with id wp-external-links-css
|
| 239 |
+
$content = preg_replace( '/<link ([^>]*)wp-external-links-css([^>]*)\/>[\s+]*/i', '', $content );
|
| 240 |
+
}
|
| 241 |
+
|
| 242 |
+
return $content;
|
| 243 |
+
}
|
| 244 |
+
|
| 245 |
+
/**
|
| 246 |
+
* Parse an attributes string into an array. If the string starts with a tag,
|
| 247 |
+
* then the attributes on the first tag are parsed. This parses via a manual
|
| 248 |
+
* loop and is designed to be safer than using DOMDocument.
|
| 249 |
+
*
|
| 250 |
+
* @param string|* $attrs
|
| 251 |
+
* @return array
|
| 252 |
+
*
|
| 253 |
+
* @example parse_attrs( 'src="example.jpg" alt="example"' )
|
| 254 |
+
* @example parse_attrs( '<img src="example.jpg" alt="example">' )
|
| 255 |
+
* @example parse_attrs( '<a href="example"></a>' )
|
| 256 |
+
* @example parse_attrs( '<a href="example">' )
|
| 257 |
+
*
|
| 258 |
+
* @link http://dev.airve.com/demo/speed_tests/php/parse_attrs.php
|
| 259 |
+
*/
|
| 260 |
+
private function parse_attrs ($attrs) {
|
| 261 |
+
if ( ! is_scalar($attrs) )
|
| 262 |
+
return (array) $attrs;
|
| 263 |
+
|
| 264 |
+
$attrs = str_split( trim($attrs) );
|
| 265 |
+
|
| 266 |
+
if ( '<' === $attrs[0] ) # looks like a tag so strip the tagname
|
| 267 |
+
while ( $attrs && ! ctype_space($attrs[0]) && $attrs[0] !== '>' )
|
| 268 |
+
array_shift($attrs);
|
| 269 |
+
|
| 270 |
+
$arr = array(); # output
|
| 271 |
+
$name = ''; # for the current attr being parsed
|
| 272 |
+
$value = ''; # for the current attr being parsed
|
| 273 |
+
$mode = 0; # whether current char is part of the name (-), the value (+), or neither (0)
|
| 274 |
+
$stop = false; # delimiter for the current $value being parsed
|
| 275 |
+
$space = ' '; # a single space
|
| 276 |
+
|
| 277 |
+
foreach ( $attrs as $j => $curr ) {
|
| 278 |
+
if ( $mode < 0 ) {# name
|
| 279 |
+
if ( '=' === $curr ) {
|
| 280 |
+
$mode = 1;
|
| 281 |
+
$stop = false;
|
| 282 |
+
} elseif ( '>' === $curr ) {
|
| 283 |
+
'' === $name or $arr[ $name ] = $value;
|
| 284 |
+
break;
|
| 285 |
+
} elseif ( ! ctype_space($curr) ) {
|
| 286 |
+
if ( ctype_space( $attrs[ $j - 1 ] ) ) { # previous char
|
| 287 |
+
'' === $name or $arr[ $name ] = ''; # previous name
|
| 288 |
+
$name = $curr; # initiate new
|
| 289 |
+
} else {
|
| 290 |
+
$name .= $curr;
|
| 291 |
+
}
|
| 292 |
+
}
|
| 293 |
+
} elseif ( $mode > 0 ) {# value
|
| 294 |
+
|
| 295 |
+
if ( $stop === false ) {
|
| 296 |
+
if ( ! ctype_space($curr) ) {
|
| 297 |
+
if ( '"' === $curr || "'" === $curr ) {
|
| 298 |
+
$value = '';
|
| 299 |
+
$stop = $curr;
|
| 300 |
+
} else {
|
| 301 |
+
$value = $curr;
|
| 302 |
+
$stop = $space;
|
| 303 |
+
}
|
| 304 |
+
}
|
| 305 |
+
} elseif ( $stop === $space ? ctype_space($curr) : $curr === $stop ) {
|
| 306 |
+
$arr[ $name ] = $value;
|
| 307 |
+
$mode = 0;
|
| 308 |
+
$name = $value = '';
|
| 309 |
+
} else {
|
| 310 |
+
$value .= $curr;
|
| 311 |
+
}
|
| 312 |
+
} else {# neither
|
| 313 |
+
|
| 314 |
+
if ( '>' === $curr )
|
| 315 |
+
break;
|
| 316 |
+
if ( ! ctype_space( $curr ) ) {
|
| 317 |
+
# initiate
|
| 318 |
+
$name = $curr;
|
| 319 |
+
$mode = -1;
|
| 320 |
+
}
|
| 321 |
+
}
|
| 322 |
+
}
|
| 323 |
+
|
| 324 |
+
# incl the final pair if it was quoteless
|
| 325 |
+
'' === $name or $arr[ $name ] = $value;
|
| 326 |
+
|
| 327 |
+
return $arr;
|
| 328 |
+
}
|
| 329 |
+
|
| 330 |
+
/**
|
| 331 |
+
* Make a clean <a> code (callback for regexp)
|
| 332 |
+
* @param array $matches Result of a preg call in filter_content()
|
| 333 |
+
* @return string Clean <a> code
|
| 334 |
+
*/
|
| 335 |
+
public function call_parse_link( $matches ) {
|
| 336 |
+
$link = $matches[ 0 ];
|
| 337 |
+
$label = $matches[ 2 ];
|
| 338 |
+
$created_link = $link;
|
| 339 |
+
|
| 340 |
+
// parse attributes
|
| 341 |
+
$attrs = $matches[ 1 ];
|
| 342 |
+
$attrs = stripslashes( $attrs );
|
| 343 |
+
$attrs = $this->parse_attrs( $attrs );
|
| 344 |
+
|
| 345 |
+
$rel = ( isset( $attrs[ 'rel' ] ) ) ? strtolower( $attrs[ 'rel' ] ) : '';
|
| 346 |
+
|
| 347 |
+
// href preperation
|
| 348 |
+
$href = $attrs[ 'href' ];
|
| 349 |
+
$href = strtolower( $href );
|
| 350 |
+
$href = trim( $href );
|
| 351 |
+
|
| 352 |
+
// checks
|
| 353 |
+
$is_external = $this->is_external( $href, $rel );
|
| 354 |
+
$is_ignored = $this->is_ignored( $href );
|
| 355 |
+
|
| 356 |
+
// is an internal link?
|
| 357 |
+
// rel=external will be threaded as external link
|
| 358 |
+
if ( ! $is_external && strpos( $rel, 'external' ) === FALSE) {
|
| 359 |
+
return apply_filters('wpel_internal_link', $created_link, $label, $attrs);
|
| 360 |
+
}
|
| 361 |
+
|
| 362 |
+
if ( $is_ignored ) {
|
| 363 |
+
return apply_filters('wpel_external_link', $created_link, $link, $label, $attrs, TRUE);
|
| 364 |
+
}
|
| 365 |
+
|
| 366 |
+
// set rel="external" (when not already set)
|
| 367 |
+
if ( $this->get_opt( 'external' ) )
|
| 368 |
+
$this->add_attr_value( $attrs, 'rel', 'external' );
|
| 369 |
+
|
| 370 |
+
// set rel="nofollow" when doesn't have "follow" (or already "nofollow")
|
| 371 |
+
if ( $this->get_opt( 'nofollow' ) AND strpos( $rel, 'follow' ) === FALSE )
|
| 372 |
+
$this->add_attr_value( $attrs, 'rel', 'nofollow' );
|
| 373 |
+
|
| 374 |
+
// set title
|
| 375 |
+
$title_format = $this->get_opt( 'title' );
|
| 376 |
+
$title = ( isset( $attrs[ 'title' ] ) ) ? $attrs[ 'title' ] : '';
|
| 377 |
+
$attrs[ 'title' ] = str_replace( '%title%', $title, $title_format );
|
| 378 |
+
|
| 379 |
+
// set user-defined class
|
| 380 |
+
$class = $this->get_opt( 'class_name' );
|
| 381 |
+
if ( $class )
|
| 382 |
+
$this->add_attr_value( $attrs, 'class', $class );
|
| 383 |
+
|
| 384 |
+
// set icon class, unless no-icon class isset or another icon class ('ext-icon-...') is found or content contains image
|
| 385 |
+
if ( $this->get_opt( 'icon' ) > 0
|
| 386 |
+
AND ( ! $this->get_opt( 'no_icon_class' ) OR strpos( $attrs[ 'class' ], $this->get_opt( 'no_icon_class' ) ) === FALSE )
|
| 387 |
+
AND strpos( $attrs[ 'class' ], 'ext-icon-' ) === FALSE
|
| 388 |
+
AND !( $this->get_opt( 'image_no_icon' ) AND (bool) preg_match( '/<img([^>]*)>/is', $label )) ){
|
| 389 |
+
$icon_class = 'ext-icon-'. $this->get_opt( 'icon', 'style' );
|
| 390 |
+
$this->add_attr_value( $attrs, 'class', $icon_class );
|
| 391 |
+
}
|
| 392 |
+
|
| 393 |
+
// set target
|
| 394 |
+
if ( ! $this->get_opt( 'use_js' ) AND ( ! $this->get_opt( 'no_icon_same_window' )
|
| 395 |
+
OR ! $this->get_opt( 'no_icon_class' )
|
| 396 |
+
OR strpos( $attrs[ 'class' ], $this->get_opt( 'no_icon_class' ) ) === FALSE ) ) {
|
| 397 |
+
if ( $this->get_opt( 'target' ) == '_none' ) {
|
| 398 |
+
unset( $attrs[ 'target' ] );
|
| 399 |
+
} else {
|
| 400 |
+
$attrs[ 'target' ] = $this->get_opt( 'target' );
|
| 401 |
+
}
|
| 402 |
+
}
|
| 403 |
+
|
| 404 |
+
// create element code
|
| 405 |
+
$created_link = '<a';
|
| 406 |
+
|
| 407 |
+
foreach ( $attrs AS $key => $value ) {
|
| 408 |
+
$created_link .= ' '. $key .'="'. $value .'"';
|
| 409 |
+
}
|
| 410 |
+
|
| 411 |
+
$created_link .= '>'. $label .'</a>';
|
| 412 |
+
|
| 413 |
+
// filter
|
| 414 |
+
$created_link = apply_filters('wpel_external_link', $created_link, $link, $label, $attrs, FALSE);
|
| 415 |
+
|
| 416 |
+
return $created_link;
|
| 417 |
+
}
|
| 418 |
+
|
| 419 |
+
/**
|
| 420 |
+
* Add value to attribute
|
| 421 |
+
* @param array $attrs
|
| 422 |
+
* @param string $attr
|
| 423 |
+
* @param string $value
|
| 424 |
+
* @param string $default Optional, default NULL which means tje attribute will be removed when (new) value is empty
|
| 425 |
+
* @return New value
|
| 426 |
+
*/
|
| 427 |
+
public function add_attr_value( &$attrs, $attr_name, $value, $default = NULL ) {
|
| 428 |
+
if ( key_exists( $attr_name, $attrs ) )
|
| 429 |
+
$old_value = $attrs[ $attr_name ];
|
| 430 |
+
|
| 431 |
+
if ( empty( $old_value ) )
|
| 432 |
+
$old_value = '';
|
| 433 |
+
|
| 434 |
+
$split = explode( ' ', strtolower( $old_value ) );
|
| 435 |
+
|
| 436 |
+
if ( in_array( $value, $split ) ) {
|
| 437 |
+
$value = $old_value;
|
| 438 |
+
} else {
|
| 439 |
+
$value = ( empty( $old_value ) )
|
| 440 |
+
? $value
|
| 441 |
+
: $old_value .' '. $value;
|
| 442 |
+
}
|
| 443 |
+
|
| 444 |
+
if ( empty( $value ) AND $default === NULL ) {
|
| 445 |
+
unset( $attrs[ $attr_name ] );
|
| 446 |
+
} else {
|
| 447 |
+
$attrs[ $attr_name ] = $value;
|
| 448 |
+
}
|
| 449 |
+
|
| 450 |
+
return $value;
|
| 451 |
+
}
|
| 452 |
+
|
| 453 |
+
/**
|
| 454 |
+
* Experimental phpQuery...
|
| 455 |
+
*/
|
| 456 |
+
|
| 457 |
+
/**
|
| 458 |
+
* Filter content
|
| 459 |
+
* @param string $content
|
| 460 |
+
* @return string
|
| 461 |
+
*/
|
| 462 |
+
private function filter_phpquery( $content ) {
|
| 463 |
+
// Workaround: remove <head>-attributes before using phpQuery
|
| 464 |
+
$regexp_head = '/<head(>|\s(.*?)>)>/is';
|
| 465 |
+
$clean_head = '<head>';
|
| 466 |
+
|
| 467 |
+
// set simple <head> without attributes
|
| 468 |
+
preg_match( $regexp_head, $content, $matches );
|
| 469 |
+
|
| 470 |
+
if( count( $matches ) > 0 ) {
|
| 471 |
+
$original_head = $matches[ 0 ];
|
| 472 |
+
$content = str_replace( $original_head, $clean_head, $content );
|
| 473 |
+
}
|
| 474 |
+
|
| 475 |
+
//phpQuery::$debug = true;
|
| 476 |
+
|
| 477 |
+
// set document
|
| 478 |
+
$doc = phpQuery::newDocument( $content );
|
| 479 |
+
|
| 480 |
+
$excl_sel = $this->get_opt( 'filter_excl_sel' );
|
| 481 |
+
|
| 482 |
+
// set excludes
|
| 483 |
+
if ( ! empty( $excl_sel ) ) {
|
| 484 |
+
$excludes = $doc->find( $excl_sel );
|
| 485 |
+
$excludes->filter( 'a' )->attr( 'excluded', true );
|
| 486 |
+
$excludes->find( 'a' )->attr( 'excluded', true );
|
| 487 |
+
}
|
| 488 |
+
|
| 489 |
+
// get <a>-tags
|
| 490 |
+
$links = $doc->find( 'a' );
|
| 491 |
+
|
| 492 |
+
// set links
|
| 493 |
+
$count = count( $links );
|
| 494 |
+
|
| 495 |
+
for( $x = 0; $x < $count; $x++ ) {
|
| 496 |
+
$a = $links->eq( $x );
|
| 497 |
+
|
| 498 |
+
if ( ! $a->attr( 'excluded' ) )
|
| 499 |
+
$this->set_link_phpquery( $links->eq( $x ) );
|
| 500 |
+
}
|
| 501 |
+
|
| 502 |
+
// remove excluded
|
| 503 |
+
if ( ! empty( $excl_sel ) ) {
|
| 504 |
+
$excludes = $doc->find( $excl_sel );
|
| 505 |
+
$excludes->filter( 'a' )->removeAttr( 'excluded' );
|
| 506 |
+
$excludes->find( 'a' )->removeAttr( 'excluded' );
|
| 507 |
+
}
|
| 508 |
+
|
| 509 |
+
// remove style when no icon classes are found
|
| 510 |
+
if ( strpos( $doc, 'ext-icon-' ) === FALSE ) {
|
| 511 |
+
// remove icon css
|
| 512 |
+
$css = $doc->find( 'link#wp-external-links-css' )->eq(0);
|
| 513 |
+
$css->remove();
|
| 514 |
+
}
|
| 515 |
+
|
| 516 |
+
// get document content
|
| 517 |
+
$content = (string) $doc;
|
| 518 |
+
|
| 519 |
+
if( isset( $original_head ) ) {
|
| 520 |
+
// recover original <head> with attributes
|
| 521 |
+
$content = str_replace( $clean_head, $original_head, $content );
|
| 522 |
+
}
|
| 523 |
+
|
| 524 |
+
return $content;
|
| 525 |
+
}
|
| 526 |
+
|
| 527 |
+
/**
|
| 528 |
+
* Set link...
|
| 529 |
+
* @param Node $a
|
| 530 |
+
* @return Node
|
| 531 |
+
*/
|
| 532 |
+
public function set_link_phpquery( $a ) {
|
| 533 |
+
$href = strtolower( $a->attr( 'href' ) . '' );
|
| 534 |
+
$rel = strtolower( $a->attr( 'rel' ) . '' );
|
| 535 |
+
|
| 536 |
+
// check if it is an external link and not excluded
|
| 537 |
+
if ( ! $this->is_external( $href, $rel ) || $this->is_ignored( $href ) )
|
| 538 |
+
return $a;
|
| 539 |
+
|
| 540 |
+
// add "external" to rel-attribute
|
| 541 |
+
if ( $this->get_opt( 'external' ) ){
|
| 542 |
+
$this->add_attr_value_phpquery( $a, 'rel', 'external' );
|
| 543 |
+
}
|
| 544 |
+
|
| 545 |
+
// add "nofollow" to rel-attribute, when doesn't have "follow"
|
| 546 |
+
if ( $this->get_opt( 'nofollow' ) AND strpos( $rel, 'follow' ) === FALSE ){
|
| 547 |
+
$this->add_attr_value_phpquery( $a, 'rel', 'nofollow' );
|
| 548 |
+
}
|
| 549 |
+
|
| 550 |
+
// set title
|
| 551 |
+
$title = str_replace( '%title%', $a->attr( 'title' ), $this->get_opt( 'title' ) );
|
| 552 |
+
$a->attr( 'title', $title );
|
| 553 |
+
|
| 554 |
+
// add icon class, unless no-icon class isset or another icon class ('ext-icon-...') is found
|
| 555 |
+
if ( $this->get_opt( 'icon' ) > 0 AND ( ! $this->get_opt( 'no_icon_class' ) OR strpos( $a->attr( 'class' ), $this->get_opt( 'no_icon_class' ) ) === FALSE ) AND strpos( $a->attr( 'class' ), 'ext-icon-' ) === FALSE ){
|
| 556 |
+
$icon_class = 'ext-icon-'. $this->get_opt( 'icon' );
|
| 557 |
+
$a->addClass( $icon_class );
|
| 558 |
+
}
|
| 559 |
+
|
| 560 |
+
// add user-defined class
|
| 561 |
+
if ( $this->get_opt( 'class_name' ) ){
|
| 562 |
+
$a->addClass( $this->get_opt( 'class_name' ) );
|
| 563 |
+
}
|
| 564 |
+
|
| 565 |
+
// set target
|
| 566 |
+
if ( $this->get_opt( 'target' ) != '_none' AND ! $this->get_opt( 'use_js' ) AND ( ! $this->get_opt( 'no_icon_same_window' ) OR ! $this->get_opt( 'no_icon_class' ) OR strpos( $a->attr( 'class' ), $this->get_opt( 'no_icon_class' ) ) === FALSE ) )
|
| 567 |
+
$a->attr( 'target', $this->get_opt( 'target' ) );
|
| 568 |
+
|
| 569 |
+
return $a;
|
| 570 |
+
}
|
| 571 |
+
|
| 572 |
+
/**
|
| 573 |
+
* Add value to attribute
|
| 574 |
+
* @param Node $node
|
| 575 |
+
* @param string $attr
|
| 576 |
+
* @param string $value
|
| 577 |
+
* @return New value
|
| 578 |
+
*/
|
| 579 |
+
private function add_attr_value_phpquery( $node, $attr, $value ) {
|
| 580 |
+
$old_value = $node->attr( $attr );
|
| 581 |
+
|
| 582 |
+
if ( empty( $old_value ) )
|
| 583 |
+
$old_value = '';
|
| 584 |
+
|
| 585 |
+
$split = split( ' ', strtolower( $old_value ) );
|
| 586 |
+
|
| 587 |
+
if ( in_array( $value, $split ) ) {
|
| 588 |
+
$value = $old_value;
|
| 589 |
+
} else {
|
| 590 |
+
$value = ( empty( $old_value ) )
|
| 591 |
+
? $value
|
| 592 |
+
: $old_value .' '. $value;
|
| 593 |
+
}
|
| 594 |
+
|
| 595 |
+
$node->attr( $attr, $value );
|
| 596 |
+
|
| 597 |
+
return $value;
|
| 598 |
+
}
|
| 599 |
+
|
| 600 |
+
} // End WP_External_Links Class
|
| 601 |
+
|
| 602 |
+
endif;
|
| 603 |
+
|
| 604 |
/* ommit PHP closing tag, to prevent unwanted whitespace at the end of the parts generated by the included files */
|
includes/phpQuery.php
CHANGED
|
@@ -1,5702 +1,5702 @@
|
|
| 1 |
-
<?php
|
| 2 |
-
/**
|
| 3 |
-
* phpQuery is a server-side, chainable, CSS3 selector driven
|
| 4 |
-
* Document Object Model (DOM) API based on jQuery JavaScript Library.
|
| 5 |
-
*
|
| 6 |
-
* @version 0.9.5
|
| 7 |
-
* @link http://code.google.com/p/phpquery/
|
| 8 |
-
* @link http://phpquery-library.blogspot.com/
|
| 9 |
-
* @link http://jquery.com/
|
| 10 |
-
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 11 |
-
* @license http://www.opensource.org/licenses/mit-license.php MIT License
|
| 12 |
-
* @package phpQuery
|
| 13 |
-
*/
|
| 14 |
-
|
| 15 |
-
// class names for instanceof
|
| 16 |
-
// TODO move them as class constants into phpQuery
|
| 17 |
-
define('DOMDOCUMENT', 'DOMDocument');
|
| 18 |
-
define('DOMELEMENT', 'DOMElement');
|
| 19 |
-
define('DOMNODELIST', 'DOMNodeList');
|
| 20 |
-
define('DOMNODE', 'DOMNode');
|
| 21 |
-
|
| 22 |
-
/**
|
| 23 |
-
* DOMEvent class.
|
| 24 |
-
*
|
| 25 |
-
* Based on
|
| 26 |
-
* @link http://developer.mozilla.org/En/DOM:event
|
| 27 |
-
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 28 |
-
* @package phpQuery
|
| 29 |
-
* @todo implement ArrayAccess ?
|
| 30 |
-
*/
|
| 31 |
-
class DOMEvent {
|
| 32 |
-
/**
|
| 33 |
-
* Returns a boolean indicating whether the event bubbles up through the DOM or not.
|
| 34 |
-
*
|
| 35 |
-
* @var unknown_type
|
| 36 |
-
*/
|
| 37 |
-
public $bubbles = true;
|
| 38 |
-
/**
|
| 39 |
-
* Returns a boolean indicating whether the event is cancelable.
|
| 40 |
-
*
|
| 41 |
-
* @var unknown_type
|
| 42 |
-
*/
|
| 43 |
-
public $cancelable = true;
|
| 44 |
-
/**
|
| 45 |
-
* Returns a reference to the currently registered target for the event.
|
| 46 |
-
*
|
| 47 |
-
* @var unknown_type
|
| 48 |
-
*/
|
| 49 |
-
public $currentTarget;
|
| 50 |
-
/**
|
| 51 |
-
* Returns detail about the event, depending on the type of event.
|
| 52 |
-
*
|
| 53 |
-
* @var unknown_type
|
| 54 |
-
* @link http://developer.mozilla.org/en/DOM/event.detail
|
| 55 |
-
*/
|
| 56 |
-
public $detail; // ???
|
| 57 |
-
/**
|
| 58 |
-
* Used to indicate which phase of the event flow is currently being evaluated.
|
| 59 |
-
*
|
| 60 |
-
* NOT IMPLEMENTED
|
| 61 |
-
*
|
| 62 |
-
* @var unknown_type
|
| 63 |
-
* @link http://developer.mozilla.org/en/DOM/event.eventPhase
|
| 64 |
-
*/
|
| 65 |
-
public $eventPhase; // ???
|
| 66 |
-
/**
|
| 67 |
-
* The explicit original target of the event (Mozilla-specific).
|
| 68 |
-
*
|
| 69 |
-
* NOT IMPLEMENTED
|
| 70 |
-
*
|
| 71 |
-
* @var unknown_type
|
| 72 |
-
*/
|
| 73 |
-
public $explicitOriginalTarget; // moz only
|
| 74 |
-
/**
|
| 75 |
-
* The original target of the event, before any retargetings (Mozilla-specific).
|
| 76 |
-
*
|
| 77 |
-
* NOT IMPLEMENTED
|
| 78 |
-
*
|
| 79 |
-
* @var unknown_type
|
| 80 |
-
*/
|
| 81 |
-
public $originalTarget; // moz only
|
| 82 |
-
/**
|
| 83 |
-
* Identifies a secondary target for the event.
|
| 84 |
-
*
|
| 85 |
-
* @var unknown_type
|
| 86 |
-
*/
|
| 87 |
-
public $relatedTarget;
|
| 88 |
-
/**
|
| 89 |
-
* Returns a reference to the target to which the event was originally dispatched.
|
| 90 |
-
*
|
| 91 |
-
* @var unknown_type
|
| 92 |
-
*/
|
| 93 |
-
public $target;
|
| 94 |
-
/**
|
| 95 |
-
* Returns the time that the event was created.
|
| 96 |
-
*
|
| 97 |
-
* @var unknown_type
|
| 98 |
-
*/
|
| 99 |
-
public $timeStamp;
|
| 100 |
-
/**
|
| 101 |
-
* Returns the name of the event (case-insensitive).
|
| 102 |
-
*/
|
| 103 |
-
public $type;
|
| 104 |
-
public $runDefault = true;
|
| 105 |
-
public $data = null;
|
| 106 |
-
public function __construct($data) {
|
| 107 |
-
foreach($data as $k => $v) {
|
| 108 |
-
$this->$k = $v;
|
| 109 |
-
}
|
| 110 |
-
if (! $this->timeStamp)
|
| 111 |
-
$this->timeStamp = time();
|
| 112 |
-
}
|
| 113 |
-
/**
|
| 114 |
-
* Cancels the event (if it is cancelable).
|
| 115 |
-
*
|
| 116 |
-
*/
|
| 117 |
-
public function preventDefault() {
|
| 118 |
-
$this->runDefault = false;
|
| 119 |
-
}
|
| 120 |
-
/**
|
| 121 |
-
* Stops the propagation of events further along in the DOM.
|
| 122 |
-
*
|
| 123 |
-
*/
|
| 124 |
-
public function stopPropagation() {
|
| 125 |
-
$this->bubbles = false;
|
| 126 |
-
}
|
| 127 |
-
}
|
| 128 |
-
|
| 129 |
-
|
| 130 |
-
/**
|
| 131 |
-
* DOMDocumentWrapper class simplifies work with DOMDocument.
|
| 132 |
-
*
|
| 133 |
-
* Know bug:
|
| 134 |
-
* - in XHTML fragments, <br /> changes to <br clear="none" />
|
| 135 |
-
*
|
| 136 |
-
* @todo check XML catalogs compatibility
|
| 137 |
-
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 138 |
-
* @package phpQuery
|
| 139 |
-
*/
|
| 140 |
-
class DOMDocumentWrapper {
|
| 141 |
-
/**
|
| 142 |
-
* @var DOMDocument
|
| 143 |
-
*/
|
| 144 |
-
public $document;
|
| 145 |
-
public $id;
|
| 146 |
-
/**
|
| 147 |
-
* @todo Rewrite as method and quess if null.
|
| 148 |
-
* @var unknown_type
|
| 149 |
-
*/
|
| 150 |
-
public $contentType = '';
|
| 151 |
-
public $xpath;
|
| 152 |
-
public $uuid = 0;
|
| 153 |
-
public $data = array();
|
| 154 |
-
public $dataNodes = array();
|
| 155 |
-
public $events = array();
|
| 156 |
-
public $eventsNodes = array();
|
| 157 |
-
public $eventsGlobal = array();
|
| 158 |
-
/**
|
| 159 |
-
* @TODO iframes support http://code.google.com/p/phpquery/issues/detail?id=28
|
| 160 |
-
* @var unknown_type
|
| 161 |
-
*/
|
| 162 |
-
public $frames = array();
|
| 163 |
-
/**
|
| 164 |
-
* Document root, by default equals to document itself.
|
| 165 |
-
* Used by documentFragments.
|
| 166 |
-
*
|
| 167 |
-
* @var DOMNode
|
| 168 |
-
*/
|
| 169 |
-
public $root;
|
| 170 |
-
public $isDocumentFragment;
|
| 171 |
-
public $isXML = false;
|
| 172 |
-
public $isXHTML = false;
|
| 173 |
-
public $isHTML = false;
|
| 174 |
-
public $charset;
|
| 175 |
-
public function __construct($markup = null, $contentType = null, $newDocumentID = null) {
|
| 176 |
-
if (isset($markup))
|
| 177 |
-
$this->load($markup, $contentType, $newDocumentID);
|
| 178 |
-
$this->id = $newDocumentID
|
| 179 |
-
? $newDocumentID
|
| 180 |
-
: md5(microtime());
|
| 181 |
-
}
|
| 182 |
-
public function load($markup, $contentType = null, $newDocumentID = null) {
|
| 183 |
-
// phpQuery::$documents[$id] = $this;
|
| 184 |
-
$this->contentType = strtolower($contentType);
|
| 185 |
-
if ($markup instanceof DOMDOCUMENT) {
|
| 186 |
-
$this->document = $markup;
|
| 187 |
-
$this->root = $this->document;
|
| 188 |
-
$this->charset = $this->document->encoding;
|
| 189 |
-
// TODO isDocumentFragment
|
| 190 |
-
} else {
|
| 191 |
-
$loaded = $this->loadMarkup($markup);
|
| 192 |
-
}
|
| 193 |
-
if ($loaded) {
|
| 194 |
-
// $this->document->formatOutput = true;
|
| 195 |
-
$this->document->preserveWhiteSpace = true;
|
| 196 |
-
$this->xpath = new DOMXPath($this->document);
|
| 197 |
-
$this->afterMarkupLoad();
|
| 198 |
-
return true;
|
| 199 |
-
// remember last loaded document
|
| 200 |
-
// return phpQuery::selectDocument($id);
|
| 201 |
-
}
|
| 202 |
-
return false;
|
| 203 |
-
}
|
| 204 |
-
protected function afterMarkupLoad() {
|
| 205 |
-
if ($this->isXHTML) {
|
| 206 |
-
$this->xpath->registerNamespace("html", "http://www.w3.org/1999/xhtml");
|
| 207 |
-
}
|
| 208 |
-
}
|
| 209 |
-
protected function loadMarkup($markup) {
|
| 210 |
-
$loaded = false;
|
| 211 |
-
if ($this->contentType) {
|
| 212 |
-
self::debug("Load markup for content type {$this->contentType}");
|
| 213 |
-
// content determined by contentType
|
| 214 |
-
list($contentType, $charset) = $this->contentTypeToArray($this->contentType);
|
| 215 |
-
switch($contentType) {
|
| 216 |
-
case 'text/html':
|
| 217 |
-
phpQuery::debug("Loading HTML, content type '{$this->contentType}'");
|
| 218 |
-
$loaded = $this->loadMarkupHTML($markup, $charset);
|
| 219 |
-
break;
|
| 220 |
-
case 'text/xml':
|
| 221 |
-
case 'application/xhtml+xml':
|
| 222 |
-
phpQuery::debug("Loading XML, content type '{$this->contentType}'");
|
| 223 |
-
$loaded = $this->loadMarkupXML($markup, $charset);
|
| 224 |
-
break;
|
| 225 |
-
default:
|
| 226 |
-
// for feeds or anything that sometimes doesn't use text/xml
|
| 227 |
-
if (strpos('xml', $this->contentType) !== false) {
|
| 228 |
-
phpQuery::debug("Loading XML, content type '{$this->contentType}'");
|
| 229 |
-
$loaded = $this->loadMarkupXML($markup, $charset);
|
| 230 |
-
} else
|
| 231 |
-
phpQuery::debug("Could not determine document type from content type '{$this->contentType}'");
|
| 232 |
-
}
|
| 233 |
-
} else {
|
| 234 |
-
// content type autodetection
|
| 235 |
-
if ($this->isXML($markup)) {
|
| 236 |
-
phpQuery::debug("Loading XML, isXML() == true");
|
| 237 |
-
$loaded = $this->loadMarkupXML($markup);
|
| 238 |
-
if (! $loaded && $this->isXHTML) {
|
| 239 |
-
phpQuery::debug('Loading as XML failed, trying to load as HTML, isXHTML == true');
|
| 240 |
-
$loaded = $this->loadMarkupHTML($markup);
|
| 241 |
-
}
|
| 242 |
-
} else {
|
| 243 |
-
phpQuery::debug("Loading HTML, isXML() == false");
|
| 244 |
-
$loaded = $this->loadMarkupHTML($markup);
|
| 245 |
-
}
|
| 246 |
-
}
|
| 247 |
-
return $loaded;
|
| 248 |
-
}
|
| 249 |
-
protected function loadMarkupReset() {
|
| 250 |
-
$this->isXML = $this->isXHTML = $this->isHTML = false;
|
| 251 |
-
}
|
| 252 |
-
protected function documentCreate($charset, $version = '1.0') {
|
| 253 |
-
if (! $version)
|
| 254 |
-
$version = '1.0';
|
| 255 |
-
$this->document = new DOMDocument($version, $charset);
|
| 256 |
-
$this->charset = $this->document->encoding;
|
| 257 |
-
// $this->document->encoding = $charset;
|
| 258 |
-
$this->document->formatOutput = true;
|
| 259 |
-
$this->document->preserveWhiteSpace = true;
|
| 260 |
-
}
|
| 261 |
-
protected function loadMarkupHTML($markup, $requestedCharset = null) {
|
| 262 |
-
if (phpQuery::$debug)
|
| 263 |
-
phpQuery::debug('Full markup load (HTML): '.substr($markup, 0, 250));
|
| 264 |
-
$this->loadMarkupReset();
|
| 265 |
-
$this->isHTML = true;
|
| 266 |
-
if (!isset($this->isDocumentFragment))
|
| 267 |
-
$this->isDocumentFragment = self::isDocumentFragmentHTML($markup);
|
| 268 |
-
$charset = null;
|
| 269 |
-
$documentCharset = $this->charsetFromHTML($markup);
|
| 270 |
-
$addDocumentCharset = false;
|
| 271 |
-
if ($documentCharset) {
|
| 272 |
-
$charset = $documentCharset;
|
| 273 |
-
$markup = $this->charsetFixHTML($markup);
|
| 274 |
-
} else if ($requestedCharset) {
|
| 275 |
-
$charset = $requestedCharset;
|
| 276 |
-
}
|
| 277 |
-
if (! $charset)
|
| 278 |
-
$charset = phpQuery::$defaultCharset;
|
| 279 |
-
// HTTP 1.1 says that the default charset is ISO-8859-1
|
| 280 |
-
// @see http://www.w3.org/International/O-HTTP-charset
|
| 281 |
-
if (! $documentCharset) {
|
| 282 |
-
$documentCharset = 'ISO-8859-1';
|
| 283 |
-
$addDocumentCharset = true;
|
| 284 |
-
}
|
| 285 |
-
// Should be careful here, still need 'magic encoding detection' since lots of pages have other 'default encoding'
|
| 286 |
-
// Worse, some pages can have mixed encodings... we'll try not to worry about that
|
| 287 |
-
$requestedCharset = strtoupper($requestedCharset);
|
| 288 |
-
$documentCharset = strtoupper($documentCharset);
|
| 289 |
-
phpQuery::debug("DOC: $documentCharset REQ: $requestedCharset");
|
| 290 |
-
if ($requestedCharset && $documentCharset && $requestedCharset !== $documentCharset) {
|
| 291 |
-
phpQuery::debug("CHARSET CONVERT");
|
| 292 |
-
// Document Encoding Conversion
|
| 293 |
-
// http://code.google.com/p/phpquery/issues/detail?id=86
|
| 294 |
-
if (function_exists('mb_detect_encoding')) {
|
| 295 |
-
$possibleCharsets = array($documentCharset, $requestedCharset, 'AUTO');
|
| 296 |
-
$docEncoding = mb_detect_encoding($markup, implode(', ', $possibleCharsets));
|
| 297 |
-
if (! $docEncoding)
|
| 298 |
-
$docEncoding = $documentCharset; // ok trust the document
|
| 299 |
-
phpQuery::debug("DETECTED '$docEncoding'");
|
| 300 |
-
// Detected does not match what document says...
|
| 301 |
-
if ($docEncoding !== $documentCharset) {
|
| 302 |
-
// Tricky..
|
| 303 |
-
}
|
| 304 |
-
if ($docEncoding !== $requestedCharset) {
|
| 305 |
-
phpQuery::debug("CONVERT $docEncoding => $requestedCharset");
|
| 306 |
-
$markup = mb_convert_encoding($markup, $requestedCharset, $docEncoding);
|
| 307 |
-
$markup = $this->charsetAppendToHTML($markup, $requestedCharset);
|
| 308 |
-
$charset = $requestedCharset;
|
| 309 |
-
}
|
| 310 |
-
} else {
|
| 311 |
-
phpQuery::debug("TODO: charset conversion without mbstring...");
|
| 312 |
-
}
|
| 313 |
-
}
|
| 314 |
-
$return = false;
|
| 315 |
-
if ($this->isDocumentFragment) {
|
| 316 |
-
phpQuery::debug("Full markup load (HTML), DocumentFragment detected, using charset '$charset'");
|
| 317 |
-
$return = $this->documentFragmentLoadMarkup($this, $charset, $markup);
|
| 318 |
-
} else {
|
| 319 |
-
if ($addDocumentCharset) {
|
| 320 |
-
phpQuery::debug("Full markup load (HTML), appending charset: '$charset'");
|
| 321 |
-
$markup = $this->charsetAppendToHTML($markup, $charset);
|
| 322 |
-
}
|
| 323 |
-
phpQuery::debug("Full markup load (HTML), documentCreate('$charset')");
|
| 324 |
-
$this->documentCreate($charset);
|
| 325 |
-
$return = phpQuery::$debug === 2
|
| 326 |
-
? $this->document->loadHTML($markup)
|
| 327 |
-
: @$this->document->loadHTML($markup);
|
| 328 |
-
if ($return)
|
| 329 |
-
$this->root = $this->document;
|
| 330 |
-
}
|
| 331 |
-
if ($return && ! $this->contentType)
|
| 332 |
-
$this->contentType = 'text/html';
|
| 333 |
-
return $return;
|
| 334 |
-
}
|
| 335 |
-
protected function loadMarkupXML($markup, $requestedCharset = null) {
|
| 336 |
-
if (phpQuery::$debug)
|
| 337 |
-
phpQuery::debug('Full markup load (XML): '.substr($markup, 0, 250));
|
| 338 |
-
$this->loadMarkupReset();
|
| 339 |
-
$this->isXML = true;
|
| 340 |
-
// check agains XHTML in contentType or markup
|
| 341 |
-
$isContentTypeXHTML = $this->isXHTML();
|
| 342 |
-
$isMarkupXHTML = $this->isXHTML($markup);
|
| 343 |
-
if ($isContentTypeXHTML || $isMarkupXHTML) {
|
| 344 |
-
self::debug('Full markup load (XML), XHTML detected');
|
| 345 |
-
$this->isXHTML = true;
|
| 346 |
-
}
|
| 347 |
-
// determine document fragment
|
| 348 |
-
if (! isset($this->isDocumentFragment))
|
| 349 |
-
$this->isDocumentFragment = $this->isXHTML
|
| 350 |
-
? self::isDocumentFragmentXHTML($markup)
|
| 351 |
-
: self::isDocumentFragmentXML($markup);
|
| 352 |
-
// this charset will be used
|
| 353 |
-
$charset = null;
|
| 354 |
-
// charset from XML declaration @var string
|
| 355 |
-
$documentCharset = $this->charsetFromXML($markup);
|
| 356 |
-
if (! $documentCharset) {
|
| 357 |
-
if ($this->isXHTML) {
|
| 358 |
-
// this is XHTML, try to get charset from content-type meta header
|
| 359 |
-
$documentCharset = $this->charsetFromHTML($markup);
|
| 360 |
-
if ($documentCharset) {
|
| 361 |
-
phpQuery::debug("Full markup load (XML), appending XHTML charset '$documentCharset'");
|
| 362 |
-
$this->charsetAppendToXML($markup, $documentCharset);
|
| 363 |
-
$charset = $documentCharset;
|
| 364 |
-
}
|
| 365 |
-
}
|
| 366 |
-
if (! $documentCharset) {
|
| 367 |
-
// if still no document charset...
|
| 368 |
-
$charset = $requestedCharset;
|
| 369 |
-
}
|
| 370 |
-
} else if ($requestedCharset) {
|
| 371 |
-
$charset = $requestedCharset;
|
| 372 |
-
}
|
| 373 |
-
if (! $charset) {
|
| 374 |
-
$charset = phpQuery::$defaultCharset;
|
| 375 |
-
}
|
| 376 |
-
if ($requestedCharset && $documentCharset && $requestedCharset != $documentCharset) {
|
| 377 |
-
// TODO place for charset conversion
|
| 378 |
-
// $charset = $requestedCharset;
|
| 379 |
-
}
|
| 380 |
-
$return = false;
|
| 381 |
-
if ($this->isDocumentFragment) {
|
| 382 |
-
phpQuery::debug("Full markup load (XML), DocumentFragment detected, using charset '$charset'");
|
| 383 |
-
$return = $this->documentFragmentLoadMarkup($this, $charset, $markup);
|
| 384 |
-
} else {
|
| 385 |
-
// FIXME ???
|
| 386 |
-
if ($isContentTypeXHTML && ! $isMarkupXHTML)
|
| 387 |
-
if (! $documentCharset) {
|
| 388 |
-
phpQuery::debug("Full markup load (XML), appending charset '$charset'");
|
| 389 |
-
$markup = $this->charsetAppendToXML($markup, $charset);
|
| 390 |
-
}
|
| 391 |
-
// see http://pl2.php.net/manual/en/book.dom.php#78929
|
| 392 |
-
// LIBXML_DTDLOAD (>= PHP 5.1)
|
| 393 |
-
// does XML ctalogues works with LIBXML_NONET
|
| 394 |
-
// $this->document->resolveExternals = true;
|
| 395 |
-
// TODO test LIBXML_COMPACT for performance improvement
|
| 396 |
-
// create document
|
| 397 |
-
$this->documentCreate($charset);
|
| 398 |
-
if (phpversion() < 5.1) {
|
| 399 |
-
$this->document->resolveExternals = true;
|
| 400 |
-
$return = phpQuery::$debug === 2
|
| 401 |
-
? $this->document->loadXML($markup)
|
| 402 |
-
: @$this->document->loadXML($markup);
|
| 403 |
-
} else {
|
| 404 |
-
/** @link http://pl2.php.net/manual/en/libxml.constants.php */
|
| 405 |
-
$libxmlStatic = phpQuery::$debug === 2
|
| 406 |
-
? LIBXML_DTDLOAD|LIBXML_DTDATTR|LIBXML_NONET
|
| 407 |
-
: LIBXML_DTDLOAD|LIBXML_DTDATTR|LIBXML_NONET|LIBXML_NOWARNING|LIBXML_NOERROR;
|
| 408 |
-
$return = $this->document->loadXML($markup, $libxmlStatic);
|
| 409 |
-
// if (! $return)
|
| 410 |
-
// $return = $this->document->loadHTML($markup);
|
| 411 |
-
}
|
| 412 |
-
if ($return)
|
| 413 |
-
$this->root = $this->document;
|
| 414 |
-
}
|
| 415 |
-
if ($return) {
|
| 416 |
-
if (! $this->contentType) {
|
| 417 |
-
if ($this->isXHTML)
|
| 418 |
-
$this->contentType = 'application/xhtml+xml';
|
| 419 |
-
else
|
| 420 |
-
$this->contentType = 'text/xml';
|
| 421 |
-
}
|
| 422 |
-
return $return;
|
| 423 |
-
} else {
|
| 424 |
-
throw new Exception("Error loading XML markup");
|
| 425 |
-
}
|
| 426 |
-
}
|
| 427 |
-
protected function isXHTML($markup = null) {
|
| 428 |
-
if (! isset($markup)) {
|
| 429 |
-
return strpos($this->contentType, 'xhtml') !== false;
|
| 430 |
-
}
|
| 431 |
-
// XXX ok ?
|
| 432 |
-
return strpos($markup, "<!DOCTYPE html") !== false;
|
| 433 |
-
// return stripos($doctype, 'xhtml') !== false;
|
| 434 |
-
// $doctype = isset($dom->doctype) && is_object($dom->doctype)
|
| 435 |
-
// ? $dom->doctype->publicId
|
| 436 |
-
// : self::$defaultDoctype;
|
| 437 |
-
}
|
| 438 |
-
protected function isXML($markup) {
|
| 439 |
-
// return strpos($markup, '<?xml') !== false && stripos($markup, 'xhtml') === false;
|
| 440 |
-
return strpos(substr($markup, 0, 100), '<'.'?xml') !== false;
|
| 441 |
-
}
|
| 442 |
-
protected function contentTypeToArray($contentType) {
|
| 443 |
-
$matches = explode(';', trim(strtolower($contentType)));
|
| 444 |
-
if (isset($matches[1])) {
|
| 445 |
-
$matches[1] = explode('=', $matches[1]);
|
| 446 |
-
// strip 'charset='
|
| 447 |
-
$matches[1] = isset($matches[1][1]) && trim($matches[1][1])
|
| 448 |
-
? $matches[1][1]
|
| 449 |
-
: $matches[1][0];
|
| 450 |
-
} else
|
| 451 |
-
$matches[1] = null;
|
| 452 |
-
return $matches;
|
| 453 |
-
}
|
| 454 |
-
/**
|
| 455 |
-
*
|
| 456 |
-
* @param $markup
|
| 457 |
-
* @return array contentType, charset
|
| 458 |
-
*/
|
| 459 |
-
protected function contentTypeFromHTML($markup) {
|
| 460 |
-
$matches = array();
|
| 461 |
-
// find meta tag
|
| 462 |
-
preg_match('@<meta[^>]+http-equiv\\s*=\\s*(["|\'])Content-Type\\1([^>]+?)>@i',
|
| 463 |
-
$markup, $matches
|
| 464 |
-
);
|
| 465 |
-
if (! isset($matches[0]))
|
| 466 |
-
return array(null, null);
|
| 467 |
-
// get attr 'content'
|
| 468 |
-
preg_match('@content\\s*=\\s*(["|\'])(.+?)\\1@', $matches[0], $matches);
|
| 469 |
-
if (! isset($matches[0]))
|
| 470 |
-
return array(null, null);
|
| 471 |
-
return $this->contentTypeToArray($matches[2]);
|
| 472 |
-
}
|
| 473 |
-
protected function charsetFromHTML($markup) {
|
| 474 |
-
$contentType = $this->contentTypeFromHTML($markup);
|
| 475 |
-
return $contentType[1];
|
| 476 |
-
}
|
| 477 |
-
protected function charsetFromXML($markup) {
|
| 478 |
-
$matches;
|
| 479 |
-
// find declaration
|
| 480 |
-
preg_match('@<'.'?xml[^>]+encoding\\s*=\\s*(["|\'])(.*?)\\1@i',
|
| 481 |
-
$markup, $matches
|
| 482 |
-
);
|
| 483 |
-
return isset($matches[2])
|
| 484 |
-
? strtolower($matches[2])
|
| 485 |
-
: null;
|
| 486 |
-
}
|
| 487 |
-
/**
|
| 488 |
-
* Repositions meta[type=charset] at the start of head. Bypasses DOMDocument bug.
|
| 489 |
-
*
|
| 490 |
-
* @link http://code.google.com/p/phpquery/issues/detail?id=80
|
| 491 |
-
* @param $html
|
| 492 |
-
*/
|
| 493 |
-
protected function charsetFixHTML($markup) {
|
| 494 |
-
$matches = array();
|
| 495 |
-
// find meta tag
|
| 496 |
-
preg_match('@\s*<meta[^>]+http-equiv\\s*=\\s*(["|\'])Content-Type\\1([^>]+?)>@i',
|
| 497 |
-
$markup, $matches, PREG_OFFSET_CAPTURE
|
| 498 |
-
);
|
| 499 |
-
if (! isset($matches[0]))
|
| 500 |
-
return;
|
| 501 |
-
$metaContentType = $matches[0][0];
|
| 502 |
-
$markup = substr($markup, 0, $matches[0][1])
|
| 503 |
-
.substr($markup, $matches[0][1]+strlen($metaContentType));
|
| 504 |
-
$headStart = stripos($markup, '<head>');
|
| 505 |
-
$markup = substr($markup, 0, $headStart+6).$metaContentType
|
| 506 |
-
.substr($markup, $headStart+6);
|
| 507 |
-
return $markup;
|
| 508 |
-
}
|
| 509 |
-
protected function charsetAppendToHTML($html, $charset, $xhtml = false) {
|
| 510 |
-
// remove existing meta[type=content-type]
|
| 511 |
-
$html = preg_replace('@\s*<meta[^>]+http-equiv\\s*=\\s*(["|\'])Content-Type\\1([^>]+?)>@i', '', $html);
|
| 512 |
-
$meta = '<meta http-equiv="Content-Type" content="text/html;charset='
|
| 513 |
-
.$charset.'" '
|
| 514 |
-
.($xhtml ? '/' : '')
|
| 515 |
-
.'>';
|
| 516 |
-
if (strpos($html, '<head') === false) {
|
| 517 |
-
if (strpos($hltml, '<html') === false) {
|
| 518 |
-
return $meta.$html;
|
| 519 |
-
} else {
|
| 520 |
-
return preg_replace(
|
| 521 |
-
'@<html(.*?)(?(?<!\?)>)@s',
|
| 522 |
-
"<html\\1><head>{$meta}</head>",
|
| 523 |
-
$html
|
| 524 |
-
);
|
| 525 |
-
}
|
| 526 |
-
} else {
|
| 527 |
-
return preg_replace(
|
| 528 |
-
'@<head(.*?)(?(?<!\?)>)@s',
|
| 529 |
-
'<head\\1>'.$meta,
|
| 530 |
-
$html
|
| 531 |
-
);
|
| 532 |
-
}
|
| 533 |
-
}
|
| 534 |
-
protected function charsetAppendToXML($markup, $charset) {
|
| 535 |
-
$declaration = '<'.'?xml version="1.0" encoding="'.$charset.'"?'.'>';
|
| 536 |
-
return $declaration.$markup;
|
| 537 |
-
}
|
| 538 |
-
public static function isDocumentFragmentHTML($markup) {
|
| 539 |
-
return stripos($markup, '<html') === false && stripos($markup, '<!doctype') === false;
|
| 540 |
-
}
|
| 541 |
-
public static function isDocumentFragmentXML($markup) {
|
| 542 |
-
return stripos($markup, '<'.'?xml') === false;
|
| 543 |
-
}
|
| 544 |
-
public static function isDocumentFragmentXHTML($markup) {
|
| 545 |
-
return self::isDocumentFragmentHTML($markup);
|
| 546 |
-
}
|
| 547 |
-
public function importAttr($value) {
|
| 548 |
-
// TODO
|
| 549 |
-
}
|
| 550 |
-
/**
|
| 551 |
-
*
|
| 552 |
-
* @param $source
|
| 553 |
-
* @param $target
|
| 554 |
-
* @param $sourceCharset
|
| 555 |
-
* @return array Array of imported nodes.
|
| 556 |
-
*/
|
| 557 |
-
public function import($source, $sourceCharset = null) {
|
| 558 |
-
// TODO charset conversions
|
| 559 |
-
$return = array();
|
| 560 |
-
if ($source instanceof DOMNODE && !($source instanceof DOMNODELIST))
|
| 561 |
-
$source = array($source);
|
| 562 |
-
// if (is_array($source)) {
|
| 563 |
-
// foreach($source as $node) {
|
| 564 |
-
// if (is_string($node)) {
|
| 565 |
-
// // string markup
|
| 566 |
-
// $fake = $this->documentFragmentCreate($node, $sourceCharset);
|
| 567 |
-
// if ($fake === false)
|
| 568 |
-
// throw new Exception("Error loading documentFragment markup");
|
| 569 |
-
// else
|
| 570 |
-
// $return = array_merge($return,
|
| 571 |
-
// $this->import($fake->root->childNodes)
|
| 572 |
-
// );
|
| 573 |
-
// } else {
|
| 574 |
-
// $return[] = $this->document->importNode($node, true);
|
| 575 |
-
// }
|
| 576 |
-
// }
|
| 577 |
-
// return $return;
|
| 578 |
-
// } else {
|
| 579 |
-
// // string markup
|
| 580 |
-
// $fake = $this->documentFragmentCreate($source, $sourceCharset);
|
| 581 |
-
// if ($fake === false)
|
| 582 |
-
// throw new Exception("Error loading documentFragment markup");
|
| 583 |
-
// else
|
| 584 |
-
// return $this->import($fake->root->childNodes);
|
| 585 |
-
// }
|
| 586 |
-
if (is_array($source) || $source instanceof DOMNODELIST) {
|
| 587 |
-
// dom nodes
|
| 588 |
-
self::debug('Importing nodes to document');
|
| 589 |
-
foreach($source as $node)
|
| 590 |
-
$return[] = $this->document->importNode($node, true);
|
| 591 |
-
} else {
|
| 592 |
-
// string markup
|
| 593 |
-
$fake = $this->documentFragmentCreate($source, $sourceCharset);
|
| 594 |
-
if ($fake === false)
|
| 595 |
-
throw new Exception("Error loading documentFragment markup");
|
| 596 |
-
else
|
| 597 |
-
return $this->import($fake->root->childNodes);
|
| 598 |
-
}
|
| 599 |
-
return $return;
|
| 600 |
-
}
|
| 601 |
-
/**
|
| 602 |
-
* Creates new document fragment.
|
| 603 |
-
*
|
| 604 |
-
* @param $source
|
| 605 |
-
* @return DOMDocumentWrapper
|
| 606 |
-
*/
|
| 607 |
-
protected function documentFragmentCreate($source, $charset = null) {
|
| 608 |
-
$fake = new DOMDocumentWrapper();
|
| 609 |
-
$fake->contentType = $this->contentType;
|
| 610 |
-
$fake->isXML = $this->isXML;
|
| 611 |
-
$fake->isHTML = $this->isHTML;
|
| 612 |
-
$fake->isXHTML = $this->isXHTML;
|
| 613 |
-
$fake->root = $fake->document;
|
| 614 |
-
if (! $charset)
|
| 615 |
-
$charset = $this->charset;
|
| 616 |
-
// $fake->documentCreate($this->charset);
|
| 617 |
-
if ($source instanceof DOMNODE && !($source instanceof DOMNODELIST))
|
| 618 |
-
$source = array($source);
|
| 619 |
-
if (is_array($source) || $source instanceof DOMNODELIST) {
|
| 620 |
-
// dom nodes
|
| 621 |
-
// load fake document
|
| 622 |
-
if (! $this->documentFragmentLoadMarkup($fake, $charset))
|
| 623 |
-
return false;
|
| 624 |
-
$nodes = $fake->import($source);
|
| 625 |
-
foreach($nodes as $node)
|
| 626 |
-
$fake->root->appendChild($node);
|
| 627 |
-
} else {
|
| 628 |
-
// string markup
|
| 629 |
-
$this->documentFragmentLoadMarkup($fake, $charset, $source);
|
| 630 |
-
}
|
| 631 |
-
return $fake;
|
| 632 |
-
}
|
| 633 |
-
/**
|
| 634 |
-
*
|
| 635 |
-
* @param $document DOMDocumentWrapper
|
| 636 |
-
* @param $markup
|
| 637 |
-
* @return $document
|
| 638 |
-
*/
|
| 639 |
-
private function documentFragmentLoadMarkup($fragment, $charset, $markup = null) {
|
| 640 |
-
// TODO error handling
|
| 641 |
-
// TODO copy doctype
|
| 642 |
-
// tempolary turn off
|
| 643 |
-
$fragment->isDocumentFragment = false;
|
| 644 |
-
if ($fragment->isXML) {
|
| 645 |
-
if ($fragment->isXHTML) {
|
| 646 |
-
// add FAKE element to set default namespace
|
| 647 |
-
$fragment->loadMarkupXML('<?xml version="1.0" encoding="'.$charset.'"?>'
|
| 648 |
-
.'<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" '
|
| 649 |
-
.'"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">'
|
| 650 |
-
.'<fake xmlns="http://www.w3.org/1999/xhtml">'.$markup.'</fake>');
|
| 651 |
-
$fragment->root = $fragment->document->firstChild->nextSibling;
|
| 652 |
-
} else {
|
| 653 |
-
$fragment->loadMarkupXML('<?xml version="1.0" encoding="'.$charset.'"?><fake>'.$markup.'</fake>');
|
| 654 |
-
$fragment->root = $fragment->document->firstChild;
|
| 655 |
-
}
|
| 656 |
-
} else {
|
| 657 |
-
$markup2 = phpQuery::$defaultDoctype.'<html><head><meta http-equiv="Content-Type" content="text/html;charset='
|
| 658 |
-
.$charset.'"></head>';
|
| 659 |
-
$noBody = strpos($markup, '<body') === false;
|
| 660 |
-
if ($noBody)
|
| 661 |
-
$markup2 .= '<body>';
|
| 662 |
-
$markup2 .= $markup;
|
| 663 |
-
if ($noBody)
|
| 664 |
-
$markup2 .= '</body>';
|
| 665 |
-
$markup2 .= '</html>';
|
| 666 |
-
$fragment->loadMarkupHTML($markup2);
|
| 667 |
-
// TODO resolv body tag merging issue
|
| 668 |
-
$fragment->root = $noBody
|
| 669 |
-
? $fragment->document->firstChild->nextSibling->firstChild->nextSibling
|
| 670 |
-
: $fragment->document->firstChild->nextSibling->firstChild->nextSibling;
|
| 671 |
-
}
|
| 672 |
-
if (! $fragment->root)
|
| 673 |
-
return false;
|
| 674 |
-
$fragment->isDocumentFragment = true;
|
| 675 |
-
return true;
|
| 676 |
-
}
|
| 677 |
-
protected function documentFragmentToMarkup($fragment) {
|
| 678 |
-
phpQuery::debug('documentFragmentToMarkup');
|
| 679 |
-
$tmp = $fragment->isDocumentFragment;
|
| 680 |
-
$fragment->isDocumentFragment = false;
|
| 681 |
-
$markup = $fragment->markup();
|
| 682 |
-
if ($fragment->isXML) {
|
| 683 |
-
$markup = substr($markup, 0, strrpos($markup, '</fake>'));
|
| 684 |
-
if ($fragment->isXHTML) {
|
| 685 |
-
$markup = substr($markup, strpos($markup, '<fake')+43);
|
| 686 |
-
} else {
|
| 687 |
-
$markup = substr($markup, strpos($markup, '<fake>')+6);
|
| 688 |
-
}
|
| 689 |
-
} else {
|
| 690 |
-
$markup = substr($markup, strpos($markup, '<body>')+6);
|
| 691 |
-
$markup = substr($markup, 0, strrpos($markup, '</body>'));
|
| 692 |
-
}
|
| 693 |
-
$fragment->isDocumentFragment = $tmp;
|
| 694 |
-
if (phpQuery::$debug)
|
| 695 |
-
phpQuery::debug('documentFragmentToMarkup: '.substr($markup, 0, 150));
|
| 696 |
-
return $markup;
|
| 697 |
-
}
|
| 698 |
-
/**
|
| 699 |
-
* Return document markup, starting with optional $nodes as root.
|
| 700 |
-
*
|
| 701 |
-
* @param $nodes DOMNode|DOMNodeList
|
| 702 |
-
* @return string
|
| 703 |
-
*/
|
| 704 |
-
public function markup($nodes = null, $innerMarkup = false) {
|
| 705 |
-
if (isset($nodes) && count($nodes) == 1 && $nodes[0] instanceof DOMDOCUMENT)
|
| 706 |
-
$nodes = null;
|
| 707 |
-
if (isset($nodes)) {
|
| 708 |
-
$markup = '';
|
| 709 |
-
if (!is_array($nodes) && !($nodes instanceof DOMNODELIST) )
|
| 710 |
-
$nodes = array($nodes);
|
| 711 |
-
if ($this->isDocumentFragment && ! $innerMarkup)
|
| 712 |
-
foreach($nodes as $i => $node)
|
| 713 |
-
if ($node->isSameNode($this->root)) {
|
| 714 |
-
// var_dump($node);
|
| 715 |
-
$nodes = array_slice($nodes, 0, $i)
|
| 716 |
-
+ phpQuery::DOMNodeListToArray($node->childNodes)
|
| 717 |
-
+ array_slice($nodes, $i+1);
|
| 718 |
-
}
|
| 719 |
-
if ($this->isXML && ! $innerMarkup) {
|
| 720 |
-
self::debug("Getting outerXML with charset '{$this->charset}'");
|
| 721 |
-
// we need outerXML, so we can benefit from
|
| 722 |
-
// $node param support in saveXML()
|
| 723 |
-
foreach($nodes as $node)
|
| 724 |
-
$markup .= $this->document->saveXML($node);
|
| 725 |
-
} else {
|
| 726 |
-
$loop = array();
|
| 727 |
-
if ($innerMarkup)
|
| 728 |
-
foreach($nodes as $node) {
|
| 729 |
-
if ($node->childNodes)
|
| 730 |
-
foreach($node->childNodes as $child)
|
| 731 |
-
$loop[] = $child;
|
| 732 |
-
else
|
| 733 |
-
$loop[] = $node;
|
| 734 |
-
}
|
| 735 |
-
else
|
| 736 |
-
$loop = $nodes;
|
| 737 |
-
self::debug("Getting markup, moving selected nodes (".count($loop).") to new DocumentFragment");
|
| 738 |
-
$fake = $this->documentFragmentCreate($loop);
|
| 739 |
-
$markup = $this->documentFragmentToMarkup($fake);
|
| 740 |
-
}
|
| 741 |
-
if ($this->isXHTML) {
|
| 742 |
-
self::debug("Fixing XHTML");
|
| 743 |
-
$markup = self::markupFixXHTML($markup);
|
| 744 |
-
}
|
| 745 |
-
self::debug("Markup: ".substr($markup, 0, 250));
|
| 746 |
-
return $markup;
|
| 747 |
-
} else {
|
| 748 |
-
if ($this->isDocumentFragment) {
|
| 749 |
-
// documentFragment, html only...
|
| 750 |
-
self::debug("Getting markup, DocumentFragment detected");
|
| 751 |
-
// return $this->markup(
|
| 752 |
-
//// $this->document->getElementsByTagName('body')->item(0)
|
| 753 |
-
// $this->document->root, true
|
| 754 |
-
// );
|
| 755 |
-
$markup = $this->documentFragmentToMarkup($this);
|
| 756 |
-
// no need for markupFixXHTML, as it's done thought markup($nodes) method
|
| 757 |
-
return $markup;
|
| 758 |
-
} else {
|
| 759 |
-
self::debug("Getting markup (".($this->isXML?'XML':'HTML')."), final with charset '{$this->charset}'");
|
| 760 |
-
$markup = $this->isXML
|
| 761 |
-
? $this->document->saveXML()
|
| 762 |
-
: $this->document->saveHTML();
|
| 763 |
-
if ($this->isXHTML) {
|
| 764 |
-
self::debug("Fixing XHTML");
|
| 765 |
-
$markup = self::markupFixXHTML($markup);
|
| 766 |
-
}
|
| 767 |
-
self::debug("Markup: ".substr($markup, 0, 250));
|
| 768 |
-
return $markup;
|
| 769 |
-
}
|
| 770 |
-
}
|
| 771 |
-
}
|
| 772 |
-
protected static function markupFixXHTML($markup) {
|
| 773 |
-
$markup = self::expandEmptyTag('script', $markup);
|
| 774 |
-
$markup = self::expandEmptyTag('select', $markup);
|
| 775 |
-
$markup = self::expandEmptyTag('textarea', $markup);
|
| 776 |
-
return $markup;
|
| 777 |
-
}
|
| 778 |
-
public static function debug($text) {
|
| 779 |
-
phpQuery::debug($text);
|
| 780 |
-
}
|
| 781 |
-
/**
|
| 782 |
-
* expandEmptyTag
|
| 783 |
-
*
|
| 784 |
-
* @param $tag
|
| 785 |
-
* @param $xml
|
| 786 |
-
* @return unknown_type
|
| 787 |
-
* @author mjaque at ilkebenson dot com
|
| 788 |
-
* @link http://php.net/manual/en/domdocument.savehtml.php#81256
|
| 789 |
-
*/
|
| 790 |
-
public static function expandEmptyTag($tag, $xml){
|
| 791 |
-
$indice = 0;
|
| 792 |
-
while ($indice< strlen($xml)){
|
| 793 |
-
$pos = strpos($xml, "<$tag ", $indice);
|
| 794 |
-
if ($pos){
|
| 795 |
-
$posCierre = strpos($xml, ">", $pos);
|
| 796 |
-
if ($xml[$posCierre-1] == "/"){
|
| 797 |
-
$xml = substr_replace($xml, "></$tag>", $posCierre-1, 2);
|
| 798 |
-
}
|
| 799 |
-
$indice = $posCierre;
|
| 800 |
-
}
|
| 801 |
-
else break;
|
| 802 |
-
}
|
| 803 |
-
return $xml;
|
| 804 |
-
}
|
| 805 |
-
}
|
| 806 |
-
|
| 807 |
-
/**
|
| 808 |
-
* Event handling class.
|
| 809 |
-
*
|
| 810 |
-
* @author Tobiasz Cudnik
|
| 811 |
-
* @package phpQuery
|
| 812 |
-
* @static
|
| 813 |
-
*/
|
| 814 |
-
abstract class phpQueryEvents {
|
| 815 |
-
/**
|
| 816 |
-
* Trigger a type of event on every matched element.
|
| 817 |
-
*
|
| 818 |
-
* @param DOMNode|phpQueryObject|string $document
|
| 819 |
-
* @param unknown_type $type
|
| 820 |
-
* @param unknown_type $data
|
| 821 |
-
*
|
| 822 |
-
* @TODO exclusive events (with !)
|
| 823 |
-
* @TODO global events (test)
|
| 824 |
-
* @TODO support more than event in $type (space-separated)
|
| 825 |
-
*/
|
| 826 |
-
public static function trigger($document, $type, $data = array(), $node = null) {
|
| 827 |
-
// trigger: function(type, data, elem, donative, extra) {
|
| 828 |
-
$documentID = phpQuery::getDocumentID($document);
|
| 829 |
-
$namespace = null;
|
| 830 |
-
if (strpos($type, '.') !== false)
|
| 831 |
-
list($name, $namespace) = explode('.', $type);
|
| 832 |
-
else
|
| 833 |
-
$name = $type;
|
| 834 |
-
if (! $node) {
|
| 835 |
-
if (self::issetGlobal($documentID, $type)) {
|
| 836 |
-
$pq = phpQuery::getDocument($documentID);
|
| 837 |
-
// TODO check add($pq->document)
|
| 838 |
-
$pq->find('*')->add($pq->document)
|
| 839 |
-
->trigger($type, $data);
|
| 840 |
-
}
|
| 841 |
-
} else {
|
| 842 |
-
if (isset($data[0]) && $data[0] instanceof DOMEvent) {
|
| 843 |
-
$event = $data[0];
|
| 844 |
-
$event->relatedTarget = $event->target;
|
| 845 |
-
$event->target = $node;
|
| 846 |
-
$data = array_slice($data, 1);
|
| 847 |
-
} else {
|
| 848 |
-
$event = new DOMEvent(array(
|
| 849 |
-
'type' => $type,
|
| 850 |
-
'target' => $node,
|
| 851 |
-
'timeStamp' => time(),
|
| 852 |
-
));
|
| 853 |
-
}
|
| 854 |
-
$i = 0;
|
| 855 |
-
while($node) {
|
| 856 |
-
// TODO whois
|
| 857 |
-
phpQuery::debug("Triggering ".($i?"bubbled ":'')."event '{$type}' on "
|
| 858 |
-
."node \n");//.phpQueryObject::whois($node)."\n");
|
| 859 |
-
$event->currentTarget = $node;
|
| 860 |
-
$eventNode = self::getNode($documentID, $node);
|
| 861 |
-
if (isset($eventNode->eventHandlers)) {
|
| 862 |
-
foreach($eventNode->eventHandlers as $eventType => $handlers) {
|
| 863 |
-
$eventNamespace = null;
|
| 864 |
-
if (strpos($type, '.') !== false)
|
| 865 |
-
list($eventName, $eventNamespace) = explode('.', $eventType);
|
| 866 |
-
else
|
| 867 |
-
$eventName = $eventType;
|
| 868 |
-
if ($name != $eventName)
|
| 869 |
-
continue;
|
| 870 |
-
if ($namespace && $eventNamespace && $namespace != $eventNamespace)
|
| 871 |
-
continue;
|
| 872 |
-
foreach($handlers as $handler) {
|
| 873 |
-
phpQuery::debug("Calling event handler\n");
|
| 874 |
-
$event->data = $handler['data']
|
| 875 |
-
? $handler['data']
|
| 876 |
-
: null;
|
| 877 |
-
$params = array_merge(array($event), $data);
|
| 878 |
-
$return = phpQuery::callbackRun($handler['callback'], $params);
|
| 879 |
-
if ($return === false) {
|
| 880 |
-
$event->bubbles = false;
|
| 881 |
-
}
|
| 882 |
-
}
|
| 883 |
-
}
|
| 884 |
-
}
|
| 885 |
-
// to bubble or not to bubble...
|
| 886 |
-
if (! $event->bubbles)
|
| 887 |
-
break;
|
| 888 |
-
$node = $node->parentNode;
|
| 889 |
-
$i++;
|
| 890 |
-
}
|
| 891 |
-
}
|
| 892 |
-
}
|
| 893 |
-
/**
|
| 894 |
-
* Binds a handler to one or more events (like click) for each matched element.
|
| 895 |
-
* Can also bind custom events.
|
| 896 |
-
*
|
| 897 |
-
* @param DOMNode|phpQueryObject|string $document
|
| 898 |
-
* @param unknown_type $type
|
| 899 |
-
* @param unknown_type $data Optional
|
| 900 |
-
* @param unknown_type $callback
|
| 901 |
-
*
|
| 902 |
-
* @TODO support '!' (exclusive) events
|
| 903 |
-
* @TODO support more than event in $type (space-separated)
|
| 904 |
-
* @TODO support binding to global events
|
| 905 |
-
*/
|
| 906 |
-
public static function add($document, $node, $type, $data, $callback = null) {
|
| 907 |
-
phpQuery::debug("Binding '$type' event");
|
| 908 |
-
$documentID = phpQuery::getDocumentID($document);
|
| 909 |
-
// if (is_null($callback) && is_callable($data)) {
|
| 910 |
-
// $callback = $data;
|
| 911 |
-
// $data = null;
|
| 912 |
-
// }
|
| 913 |
-
$eventNode = self::getNode($documentID, $node);
|
| 914 |
-
if (! $eventNode)
|
| 915 |
-
$eventNode = self::setNode($documentID, $node);
|
| 916 |
-
if (!isset($eventNode->eventHandlers[$type]))
|
| 917 |
-
$eventNode->eventHandlers[$type] = array();
|
| 918 |
-
$eventNode->eventHandlers[$type][] = array(
|
| 919 |
-
'callback' => $callback,
|
| 920 |
-
'data' => $data,
|
| 921 |
-
);
|
| 922 |
-
}
|
| 923 |
-
/**
|
| 924 |
-
* Enter description here...
|
| 925 |
-
*
|
| 926 |
-
* @param DOMNode|phpQueryObject|string $document
|
| 927 |
-
* @param unknown_type $type
|
| 928 |
-
* @param unknown_type $callback
|
| 929 |
-
*
|
| 930 |
-
* @TODO namespace events
|
| 931 |
-
* @TODO support more than event in $type (space-separated)
|
| 932 |
-
*/
|
| 933 |
-
public static function remove($document, $node, $type = null, $callback = null) {
|
| 934 |
-
$documentID = phpQuery::getDocumentID($document);
|
| 935 |
-
$eventNode = self::getNode($documentID, $node);
|
| 936 |
-
if (is_object($eventNode) && isset($eventNode->eventHandlers[$type])) {
|
| 937 |
-
if ($callback) {
|
| 938 |
-
foreach($eventNode->eventHandlers[$type] as $k => $handler)
|
| 939 |
-
if ($handler['callback'] == $callback)
|
| 940 |
-
unset($eventNode->eventHandlers[$type][$k]);
|
| 941 |
-
} else {
|
| 942 |
-
unset($eventNode->eventHandlers[$type]);
|
| 943 |
-
}
|
| 944 |
-
}
|
| 945 |
-
}
|
| 946 |
-
protected static function getNode($documentID, $node) {
|
| 947 |
-
foreach(phpQuery::$documents[$documentID]->eventsNodes as $eventNode) {
|
| 948 |
-
if ($node->isSameNode($eventNode))
|
| 949 |
-
return $eventNode;
|
| 950 |
-
}
|
| 951 |
-
}
|
| 952 |
-
protected static function setNode($documentID, $node) {
|
| 953 |
-
phpQuery::$documents[$documentID]->eventsNodes[] = $node;
|
| 954 |
-
return phpQuery::$documents[$documentID]->eventsNodes[
|
| 955 |
-
count(phpQuery::$documents[$documentID]->eventsNodes)-1
|
| 956 |
-
];
|
| 957 |
-
}
|
| 958 |
-
protected static function issetGlobal($documentID, $type) {
|
| 959 |
-
return isset(phpQuery::$documents[$documentID])
|
| 960 |
-
? in_array($type, phpQuery::$documents[$documentID]->eventsGlobal)
|
| 961 |
-
: false;
|
| 962 |
-
}
|
| 963 |
-
}
|
| 964 |
-
|
| 965 |
-
|
| 966 |
-
interface ICallbackNamed {
|
| 967 |
-
function hasName();
|
| 968 |
-
function getName();
|
| 969 |
-
}
|
| 970 |
-
/**
|
| 971 |
-
* Callback class introduces currying-like pattern.
|
| 972 |
-
*
|
| 973 |
-
* Example:
|
| 974 |
-
* function foo($param1, $param2, $param3) {
|
| 975 |
-
* var_dump($param1, $param2, $param3);
|
| 976 |
-
* }
|
| 977 |
-
* $fooCurried = new Callback('foo',
|
| 978 |
-
* 'param1 is now statically set',
|
| 979 |
-
* new CallbackParam, new CallbackParam
|
| 980 |
-
* );
|
| 981 |
-
* phpQuery::callbackRun($fooCurried,
|
| 982 |
-
* array('param2 value', 'param3 value'
|
| 983 |
-
* );
|
| 984 |
-
*
|
| 985 |
-
* Callback class is supported in all phpQuery methods which accepts callbacks.
|
| 986 |
-
*
|
| 987 |
-
* @link http://code.google.com/p/phpquery/wiki/Callbacks#Param_Structures
|
| 988 |
-
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 989 |
-
*
|
| 990 |
-
* @TODO??? return fake forwarding function created via create_function
|
| 991 |
-
* @TODO honor paramStructure
|
| 992 |
-
*/
|
| 993 |
-
class Callback
|
| 994 |
-
implements ICallbackNamed {
|
| 995 |
-
public $callback = null;
|
| 996 |
-
public $params = null;
|
| 997 |
-
protected $name;
|
| 998 |
-
public function __construct($callback, $param1 = null, $param2 = null,
|
| 999 |
-
$param3 = null) {
|
| 1000 |
-
$params = func_get_args();
|
| 1001 |
-
$params = array_slice($params, 1);
|
| 1002 |
-
if ($callback instanceof Callback) {
|
| 1003 |
-
// TODO implement recurention
|
| 1004 |
-
} else {
|
| 1005 |
-
$this->callback = $callback;
|
| 1006 |
-
$this->params = $params;
|
| 1007 |
-
}
|
| 1008 |
-
}
|
| 1009 |
-
public function getName() {
|
| 1010 |
-
return 'Callback: '.$this->name;
|
| 1011 |
-
}
|
| 1012 |
-
public function hasName() {
|
| 1013 |
-
return isset($this->name) && $this->name;
|
| 1014 |
-
}
|
| 1015 |
-
public function setName($name) {
|
| 1016 |
-
$this->name = $name;
|
| 1017 |
-
return $this;
|
| 1018 |
-
}
|
| 1019 |
-
// TODO test me
|
| 1020 |
-
// public function addParams() {
|
| 1021 |
-
// $params = func_get_args();
|
| 1022 |
-
// return new Callback($this->callback, $this->params+$params);
|
| 1023 |
-
// }
|
| 1024 |
-
}
|
| 1025 |
-
/**
|
| 1026 |
-
* Shorthand for new Callback(create_function(...), ...);
|
| 1027 |
-
*
|
| 1028 |
-
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 1029 |
-
*/
|
| 1030 |
-
class CallbackBody extends Callback {
|
| 1031 |
-
public function __construct($paramList, $code, $param1 = null, $param2 = null,
|
| 1032 |
-
$param3 = null) {
|
| 1033 |
-
$params = func_get_args();
|
| 1034 |
-
$params = array_slice($params, 2);
|
| 1035 |
-
$this->callback = create_function($paramList, $code);
|
| 1036 |
-
$this->params = $params;
|
| 1037 |
-
}
|
| 1038 |
-
}
|
| 1039 |
-
/**
|
| 1040 |
-
* Callback type which on execution returns reference passed during creation.
|
| 1041 |
-
*
|
| 1042 |
-
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 1043 |
-
*/
|
| 1044 |
-
class CallbackReturnReference extends Callback
|
| 1045 |
-
implements ICallbackNamed {
|
| 1046 |
-
protected $reference;
|
| 1047 |
-
public function __construct(&$reference, $name = null){
|
| 1048 |
-
$this->reference =& $reference;
|
| 1049 |
-
$this->callback = array($this, 'callback');
|
| 1050 |
-
}
|
| 1051 |
-
public function callback() {
|
| 1052 |
-
return $this->reference;
|
| 1053 |
-
}
|
| 1054 |
-
public function getName() {
|
| 1055 |
-
return 'Callback: '.$this->name;
|
| 1056 |
-
}
|
| 1057 |
-
public function hasName() {
|
| 1058 |
-
return isset($this->name) && $this->name;
|
| 1059 |
-
}
|
| 1060 |
-
}
|
| 1061 |
-
/**
|
| 1062 |
-
* Callback type which on execution returns value passed during creation.
|
| 1063 |
-
*
|
| 1064 |
-
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 1065 |
-
*/
|
| 1066 |
-
class CallbackReturnValue extends Callback
|
| 1067 |
-
implements ICallbackNamed {
|
| 1068 |
-
protected $value;
|
| 1069 |
-
protected $name;
|
| 1070 |
-
public function __construct($value, $name = null){
|
| 1071 |
-
$this->value =& $value;
|
| 1072 |
-
$this->name = $name;
|
| 1073 |
-
$this->callback = array($this, 'callback');
|
| 1074 |
-
}
|
| 1075 |
-
public function callback() {
|
| 1076 |
-
return $this->value;
|
| 1077 |
-
}
|
| 1078 |
-
public function __toString() {
|
| 1079 |
-
return $this->getName();
|
| 1080 |
-
}
|
| 1081 |
-
public function getName() {
|
| 1082 |
-
return 'Callback: '.$this->name;
|
| 1083 |
-
}
|
| 1084 |
-
public function hasName() {
|
| 1085 |
-
return isset($this->name) && $this->name;
|
| 1086 |
-
}
|
| 1087 |
-
}
|
| 1088 |
-
/**
|
| 1089 |
-
* CallbackParameterToReference can be used when we don't really want a callback,
|
| 1090 |
-
* only parameter passed to it. CallbackParameterToReference takes first
|
| 1091 |
-
* parameter's value and passes it to reference.
|
| 1092 |
-
*
|
| 1093 |
-
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 1094 |
-
*/
|
| 1095 |
-
class CallbackParameterToReference extends Callback {
|
| 1096 |
-
/**
|
| 1097 |
-
* @param $reference
|
| 1098 |
-
* @TODO implement $paramIndex;
|
| 1099 |
-
* param index choose which callback param will be passed to reference
|
| 1100 |
-
*/
|
| 1101 |
-
public function __construct(&$reference){
|
| 1102 |
-
$this->callback =& $reference;
|
| 1103 |
-
}
|
| 1104 |
-
}
|
| 1105 |
-
//class CallbackReference extends Callback {
|
| 1106 |
-
// /**
|
| 1107 |
-
// *
|
| 1108 |
-
// * @param $reference
|
| 1109 |
-
// * @param $paramIndex
|
| 1110 |
-
// * @todo implement $paramIndex; param index choose which callback param will be passed to reference
|
| 1111 |
-
// */
|
| 1112 |
-
// public function __construct(&$reference, $name = null){
|
| 1113 |
-
// $this->callback =& $reference;
|
| 1114 |
-
// }
|
| 1115 |
-
//}
|
| 1116 |
-
class CallbackParam {}
|
| 1117 |
-
|
| 1118 |
-
/**
|
| 1119 |
-
* Class representing phpQuery objects.
|
| 1120 |
-
*
|
| 1121 |
-
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 1122 |
-
* @package phpQuery
|
| 1123 |
-
* @method phpQueryObject clone() clone()
|
| 1124 |
-
* @method phpQueryObject empty() empty()
|
| 1125 |
-
* @method phpQueryObject next() next($selector = null)
|
| 1126 |
-
* @method phpQueryObject prev() prev($selector = null)
|
| 1127 |
-
* @property Int $length
|
| 1128 |
-
*/
|
| 1129 |
-
class phpQueryObject
|
| 1130 |
-
implements Iterator, Countable, ArrayAccess {
|
| 1131 |
-
public $documentID = null;
|
| 1132 |
-
/**
|
| 1133 |
-
* DOMDocument class.
|
| 1134 |
-
*
|
| 1135 |
-
* @var DOMDocument
|
| 1136 |
-
*/
|
| 1137 |
-
public $document = null;
|
| 1138 |
-
public $charset = null;
|
| 1139 |
-
/**
|
| 1140 |
-
*
|
| 1141 |
-
* @var DOMDocumentWrapper
|
| 1142 |
-
*/
|
| 1143 |
-
public $documentWrapper = null;
|
| 1144 |
-
/**
|
| 1145 |
-
* XPath interface.
|
| 1146 |
-
*
|
| 1147 |
-
* @var DOMXPath
|
| 1148 |
-
*/
|
| 1149 |
-
public $xpath = null;
|
| 1150 |
-
/**
|
| 1151 |
-
* Stack of selected elements.
|
| 1152 |
-
* @TODO refactor to ->nodes
|
| 1153 |
-
* @var array
|
| 1154 |
-
*/
|
| 1155 |
-
public $elements = array();
|
| 1156 |
-
/**
|
| 1157 |
-
* @access private
|
| 1158 |
-
*/
|
| 1159 |
-
protected $elementsBackup = array();
|
| 1160 |
-
/**
|
| 1161 |
-
* @access private
|
| 1162 |
-
*/
|
| 1163 |
-
protected $previous = null;
|
| 1164 |
-
/**
|
| 1165 |
-
* @access private
|
| 1166 |
-
* @TODO deprecate
|
| 1167 |
-
*/
|
| 1168 |
-
protected $root = array();
|
| 1169 |
-
/**
|
| 1170 |
-
* Indicated if doument is just a fragment (no <html> tag).
|
| 1171 |
-
*
|
| 1172 |
-
* Every document is realy a full document, so even documentFragments can
|
| 1173 |
-
* be queried against <html>, but getDocument(id)->htmlOuter() will return
|
| 1174 |
-
* only contents of <body>.
|
| 1175 |
-
*
|
| 1176 |
-
* @var bool
|
| 1177 |
-
*/
|
| 1178 |
-
public $documentFragment = true;
|
| 1179 |
-
/**
|
| 1180 |
-
* Iterator interface helper
|
| 1181 |
-
* @access private
|
| 1182 |
-
*/
|
| 1183 |
-
protected $elementsInterator = array();
|
| 1184 |
-
/**
|
| 1185 |
-
* Iterator interface helper
|
| 1186 |
-
* @access private
|
| 1187 |
-
*/
|
| 1188 |
-
protected $valid = false;
|
| 1189 |
-
/**
|
| 1190 |
-
* Iterator interface helper
|
| 1191 |
-
* @access private
|
| 1192 |
-
*/
|
| 1193 |
-
protected $current = null;
|
| 1194 |
-
/**
|
| 1195 |
-
* Enter description here...
|
| 1196 |
-
*
|
| 1197 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 1198 |
-
*/
|
| 1199 |
-
public function __construct($documentID) {
|
| 1200 |
-
// if ($documentID instanceof self)
|
| 1201 |
-
// var_dump($documentID->getDocumentID());
|
| 1202 |
-
$id = $documentID instanceof self
|
| 1203 |
-
? $documentID->getDocumentID()
|
| 1204 |
-
: $documentID;
|
| 1205 |
-
// var_dump($id);
|
| 1206 |
-
if (! isset(phpQuery::$documents[$id] )) {
|
| 1207 |
-
// var_dump(phpQuery::$documents);
|
| 1208 |
-
throw new Exception("Document with ID '{$id}' isn't loaded. Use phpQuery::newDocument(\$html) or phpQuery::newDocumentFile(\$file) first.");
|
| 1209 |
-
}
|
| 1210 |
-
$this->documentID = $id;
|
| 1211 |
-
$this->documentWrapper =& phpQuery::$documents[$id];
|
| 1212 |
-
$this->document =& $this->documentWrapper->document;
|
| 1213 |
-
$this->xpath =& $this->documentWrapper->xpath;
|
| 1214 |
-
$this->charset =& $this->documentWrapper->charset;
|
| 1215 |
-
$this->documentFragment =& $this->documentWrapper->isDocumentFragment;
|
| 1216 |
-
// TODO check $this->DOM->documentElement;
|
| 1217 |
-
// $this->root = $this->document->documentElement;
|
| 1218 |
-
$this->root =& $this->documentWrapper->root;
|
| 1219 |
-
// $this->toRoot();
|
| 1220 |
-
$this->elements = array($this->root);
|
| 1221 |
-
}
|
| 1222 |
-
/**
|
| 1223 |
-
*
|
| 1224 |
-
* @access private
|
| 1225 |
-
* @param $attr
|
| 1226 |
-
* @return unknown_type
|
| 1227 |
-
*/
|
| 1228 |
-
public function __get($attr) {
|
| 1229 |
-
switch($attr) {
|
| 1230 |
-
// FIXME doesnt work at all ?
|
| 1231 |
-
case 'length':
|
| 1232 |
-
return $this->size();
|
| 1233 |
-
break;
|
| 1234 |
-
default:
|
| 1235 |
-
return $this->$attr;
|
| 1236 |
-
}
|
| 1237 |
-
}
|
| 1238 |
-
/**
|
| 1239 |
-
* Saves actual object to $var by reference.
|
| 1240 |
-
* Useful when need to break chain.
|
| 1241 |
-
* @param phpQueryObject $var
|
| 1242 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 1243 |
-
*/
|
| 1244 |
-
public function toReference(&$var) {
|
| 1245 |
-
return $var = $this;
|
| 1246 |
-
}
|
| 1247 |
-
public function documentFragment($state = null) {
|
| 1248 |
-
if ($state) {
|
| 1249 |
-
phpQuery::$documents[$this->getDocumentID()]['documentFragment'] = $state;
|
| 1250 |
-
return $this;
|
| 1251 |
-
}
|
| 1252 |
-
return $this->documentFragment;
|
| 1253 |
-
}
|
| 1254 |
-
/**
|
| 1255 |
-
* @access private
|
| 1256 |
-
* @TODO documentWrapper
|
| 1257 |
-
*/
|
| 1258 |
-
protected function isRoot( $node) {
|
| 1259 |
-
// return $node instanceof DOMDOCUMENT || $node->tagName == 'html';
|
| 1260 |
-
return $node instanceof DOMDOCUMENT
|
| 1261 |
-
|| ($node instanceof DOMELEMENT && $node->tagName == 'html')
|
| 1262 |
-
|| $this->root->isSameNode($node);
|
| 1263 |
-
}
|
| 1264 |
-
/**
|
| 1265 |
-
* @access private
|
| 1266 |
-
*/
|
| 1267 |
-
protected function stackIsRoot() {
|
| 1268 |
-
return $this->size() == 1 && $this->isRoot($this->elements[0]);
|
| 1269 |
-
}
|
| 1270 |
-
/**
|
| 1271 |
-
* Enter description here...
|
| 1272 |
-
* NON JQUERY METHOD
|
| 1273 |
-
*
|
| 1274 |
-
* Watch out, it doesn't creates new instance, can be reverted with end().
|
| 1275 |
-
*
|
| 1276 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 1277 |
-
*/
|
| 1278 |
-
public function toRoot() {
|
| 1279 |
-
$this->elements = array($this->root);
|
| 1280 |
-
return $this;
|
| 1281 |
-
// return $this->newInstance(array($this->root));
|
| 1282 |
-
}
|
| 1283 |
-
/**
|
| 1284 |
-
* Saves object's DocumentID to $var by reference.
|
| 1285 |
-
* <code>
|
| 1286 |
-
* $myDocumentId;
|
| 1287 |
-
* phpQuery::newDocument('<div/>')
|
| 1288 |
-
* ->getDocumentIDRef($myDocumentId)
|
| 1289 |
-
* ->find('div')->...
|
| 1290 |
-
* </code>
|
| 1291 |
-
*
|
| 1292 |
-
* @param unknown_type $domId
|
| 1293 |
-
* @see phpQuery::newDocument
|
| 1294 |
-
* @see phpQuery::newDocumentFile
|
| 1295 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 1296 |
-
*/
|
| 1297 |
-
public function getDocumentIDRef(&$documentID) {
|
| 1298 |
-
$documentID = $this->getDocumentID();
|
| 1299 |
-
return $this;
|
| 1300 |
-
}
|
| 1301 |
-
/**
|
| 1302 |
-
* Returns object with stack set to document root.
|
| 1303 |
-
*
|
| 1304 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 1305 |
-
*/
|
| 1306 |
-
public function getDocument() {
|
| 1307 |
-
return phpQuery::getDocument($this->getDocumentID());
|
| 1308 |
-
}
|
| 1309 |
-
/**
|
| 1310 |
-
*
|
| 1311 |
-
* @return DOMDocument
|
| 1312 |
-
*/
|
| 1313 |
-
public function getDOMDocument() {
|
| 1314 |
-
return $this->document;
|
| 1315 |
-
}
|
| 1316 |
-
/**
|
| 1317 |
-
* Get object's Document ID.
|
| 1318 |
-
*
|
| 1319 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 1320 |
-
*/
|
| 1321 |
-
public function getDocumentID() {
|
| 1322 |
-
return $this->documentID;
|
| 1323 |
-
}
|
| 1324 |
-
/**
|
| 1325 |
-
* Unloads whole document from memory.
|
| 1326 |
-
* CAUTION! None further operations will be possible on this document.
|
| 1327 |
-
* All objects refering to it will be useless.
|
| 1328 |
-
*
|
| 1329 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 1330 |
-
*/
|
| 1331 |
-
public function unloadDocument() {
|
| 1332 |
-
phpQuery::unloadDocuments($this->getDocumentID());
|
| 1333 |
-
}
|
| 1334 |
-
public function isHTML() {
|
| 1335 |
-
return $this->documentWrapper->isHTML;
|
| 1336 |
-
}
|
| 1337 |
-
public function isXHTML() {
|
| 1338 |
-
return $this->documentWrapper->isXHTML;
|
| 1339 |
-
}
|
| 1340 |
-
public function isXML() {
|
| 1341 |
-
return $this->documentWrapper->isXML;
|
| 1342 |
-
}
|
| 1343 |
-
/**
|
| 1344 |
-
* Enter description here...
|
| 1345 |
-
*
|
| 1346 |
-
* @link http://docs.jquery.com/Ajax/serialize
|
| 1347 |
-
* @return string
|
| 1348 |
-
*/
|
| 1349 |
-
public function serialize() {
|
| 1350 |
-
return phpQuery::param($this->serializeArray());
|
| 1351 |
-
}
|
| 1352 |
-
/**
|
| 1353 |
-
* Enter description here...
|
| 1354 |
-
*
|
| 1355 |
-
* @link http://docs.jquery.com/Ajax/serializeArray
|
| 1356 |
-
* @return array
|
| 1357 |
-
*/
|
| 1358 |
-
public function serializeArray($submit = null) {
|
| 1359 |
-
$source = $this->filter('form, input, select, textarea')
|
| 1360 |
-
->find('input, select, textarea')
|
| 1361 |
-
->andSelf()
|
| 1362 |
-
->not('form');
|
| 1363 |
-
$return = array();
|
| 1364 |
-
// $source->dumpDie();
|
| 1365 |
-
foreach($source as $input) {
|
| 1366 |
-
$input = phpQuery::pq($input);
|
| 1367 |
-
if ($input->is('[disabled]'))
|
| 1368 |
-
continue;
|
| 1369 |
-
if (!$input->is('[name]'))
|
| 1370 |
-
continue;
|
| 1371 |
-
if ($input->is('[type=checkbox]') && !$input->is('[checked]'))
|
| 1372 |
-
continue;
|
| 1373 |
-
// jquery diff
|
| 1374 |
-
if ($submit && $input->is('[type=submit]')) {
|
| 1375 |
-
if ($submit instanceof DOMELEMENT && ! $input->elements[0]->isSameNode($submit))
|
| 1376 |
-
continue;
|
| 1377 |
-
else if (is_string($submit) && $input->attr('name') != $submit)
|
| 1378 |
-
continue;
|
| 1379 |
-
}
|
| 1380 |
-
$return[] = array(
|
| 1381 |
-
'name' => $input->attr('name'),
|
| 1382 |
-
'value' => $input->val(),
|
| 1383 |
-
);
|
| 1384 |
-
}
|
| 1385 |
-
return $return;
|
| 1386 |
-
}
|
| 1387 |
-
/**
|
| 1388 |
-
* @access private
|
| 1389 |
-
*/
|
| 1390 |
-
protected function debug($in) {
|
| 1391 |
-
if (! phpQuery::$debug )
|
| 1392 |
-
return;
|
| 1393 |
-
print('<pre>');
|
| 1394 |
-
print_r($in);
|
| 1395 |
-
// file debug
|
| 1396 |
-
// file_put_contents(dirname(__FILE__).'/phpQuery.log', print_r($in, true)."\n", FILE_APPEND);
|
| 1397 |
-
// quite handy debug trace
|
| 1398 |
-
// if ( is_array($in))
|
| 1399 |
-
// print_r(array_slice(debug_backtrace(), 3));
|
| 1400 |
-
print("</pre>\n");
|
| 1401 |
-
}
|
| 1402 |
-
/**
|
| 1403 |
-
* @access private
|
| 1404 |
-
*/
|
| 1405 |
-
protected function isRegexp($pattern) {
|
| 1406 |
-
return in_array(
|
| 1407 |
-
$pattern[ mb_strlen($pattern)-1 ],
|
| 1408 |
-
array('^','*','$')
|
| 1409 |
-
);
|
| 1410 |
-
}
|
| 1411 |
-
/**
|
| 1412 |
-
* Determines if $char is really a char.
|
| 1413 |
-
*
|
| 1414 |
-
* @param string $char
|
| 1415 |
-
* @return bool
|
| 1416 |
-
* @todo rewrite me to charcode range ! ;)
|
| 1417 |
-
* @access private
|
| 1418 |
-
*/
|
| 1419 |
-
protected function isChar($char) {
|
| 1420 |
-
return extension_loaded('mbstring') && phpQuery::$mbstringSupport
|
| 1421 |
-
? mb_eregi('\w', $char)
|
| 1422 |
-
: preg_match('@\w@', $char);
|
| 1423 |
-
}
|
| 1424 |
-
/**
|
| 1425 |
-
* @access private
|
| 1426 |
-
*/
|
| 1427 |
-
protected function parseSelector($query) {
|
| 1428 |
-
// clean spaces
|
| 1429 |
-
// TODO include this inside parsing ?
|
| 1430 |
-
$query = trim(
|
| 1431 |
-
preg_replace('@\s+@', ' ',
|
| 1432 |
-
preg_replace('@\s*(>|\\+|~)\s*@', '\\1', $query)
|
| 1433 |
-
)
|
| 1434 |
-
);
|
| 1435 |
-
$queries = array(array());
|
| 1436 |
-
if (! $query)
|
| 1437 |
-
return $queries;
|
| 1438 |
-
$return =& $queries[0];
|
| 1439 |
-
$specialChars = array('>',' ');
|
| 1440 |
-
// $specialCharsMapping = array('/' => '>');
|
| 1441 |
-
$specialCharsMapping = array();
|
| 1442 |
-
$strlen = mb_strlen($query);
|
| 1443 |
-
$classChars = array('.', '-');
|
| 1444 |
-
$pseudoChars = array('-');
|
| 1445 |
-
$tagChars = array('*', '|', '-');
|
| 1446 |
-
// split multibyte string
|
| 1447 |
-
// http://code.google.com/p/phpquery/issues/detail?id=76
|
| 1448 |
-
$_query = array();
|
| 1449 |
-
for ($i=0; $i<$strlen; $i++)
|
| 1450 |
-
$_query[] = mb_substr($query, $i, 1);
|
| 1451 |
-
$query = $_query;
|
| 1452 |
-
// it works, but i dont like it...
|
| 1453 |
-
$i = 0;
|
| 1454 |
-
while( $i < $strlen) {
|
| 1455 |
-
$c = $query[$i];
|
| 1456 |
-
$tmp = '';
|
| 1457 |
-
// TAG
|
| 1458 |
-
if ($this->isChar($c) || in_array($c, $tagChars)) {
|
| 1459 |
-
while(isset($query[$i])
|
| 1460 |
-
&& ($this->isChar($query[$i]) || in_array($query[$i], $tagChars))) {
|
| 1461 |
-
$tmp .= $query[$i];
|
| 1462 |
-
$i++;
|
| 1463 |
-
}
|
| 1464 |
-
$return[] = $tmp;
|
| 1465 |
-
// IDs
|
| 1466 |
-
} else if ( $c == '#') {
|
| 1467 |
-
$i++;
|
| 1468 |
-
while( isset($query[$i]) && ($this->isChar($query[$i]) || $query[$i] == '-')) {
|
| 1469 |
-
$tmp .= $query[$i];
|
| 1470 |
-
$i++;
|
| 1471 |
-
}
|
| 1472 |
-
$return[] = '#'.$tmp;
|
| 1473 |
-
// SPECIAL CHARS
|
| 1474 |
-
} else if (in_array($c, $specialChars)) {
|
| 1475 |
-
$return[] = $c;
|
| 1476 |
-
$i++;
|
| 1477 |
-
// MAPPED SPECIAL MULTICHARS
|
| 1478 |
-
// } else if ( $c.$query[$i+1] == '//') {
|
| 1479 |
-
// $return[] = ' ';
|
| 1480 |
-
// $i = $i+2;
|
| 1481 |
-
// MAPPED SPECIAL CHARS
|
| 1482 |
-
} else if ( isset($specialCharsMapping[$c])) {
|
| 1483 |
-
$return[] = $specialCharsMapping[$c];
|
| 1484 |
-
$i++;
|
| 1485 |
-
// COMMA
|
| 1486 |
-
} else if ( $c == ',') {
|
| 1487 |
-
$queries[] = array();
|
| 1488 |
-
$return =& $queries[ count($queries)-1 ];
|
| 1489 |
-
$i++;
|
| 1490 |
-
while( isset($query[$i]) && $query[$i] == ' ')
|
| 1491 |
-
$i++;
|
| 1492 |
-
// CLASSES
|
| 1493 |
-
} else if ($c == '.') {
|
| 1494 |
-
while( isset($query[$i]) && ($this->isChar($query[$i]) || in_array($query[$i], $classChars))) {
|
| 1495 |
-
$tmp .= $query[$i];
|
| 1496 |
-
$i++;
|
| 1497 |
-
}
|
| 1498 |
-
$return[] = $tmp;
|
| 1499 |
-
// ~ General Sibling Selector
|
| 1500 |
-
} else if ($c == '~') {
|
| 1501 |
-
$spaceAllowed = true;
|
| 1502 |
-
$tmp .= $query[$i++];
|
| 1503 |
-
while( isset($query[$i])
|
| 1504 |
-
&& ($this->isChar($query[$i])
|
| 1505 |
-
|| in_array($query[$i], $classChars)
|
| 1506 |
-
|| $query[$i] == '*'
|
| 1507 |
-
|| ($query[$i] == ' ' && $spaceAllowed)
|
| 1508 |
-
)) {
|
| 1509 |
-
if ($query[$i] != ' ')
|
| 1510 |
-
$spaceAllowed = false;
|
| 1511 |
-
$tmp .= $query[$i];
|
| 1512 |
-
$i++;
|
| 1513 |
-
}
|
| 1514 |
-
$return[] = $tmp;
|
| 1515 |
-
// + Adjacent sibling selectors
|
| 1516 |
-
} else if ($c == '+') {
|
| 1517 |
-
$spaceAllowed = true;
|
| 1518 |
-
$tmp .= $query[$i++];
|
| 1519 |
-
while( isset($query[$i])
|
| 1520 |
-
&& ($this->isChar($query[$i])
|
| 1521 |
-
|| in_array($query[$i], $classChars)
|
| 1522 |
-
|| $query[$i] == '*'
|
| 1523 |
-
|| ($spaceAllowed && $query[$i] == ' ')
|
| 1524 |
-
)) {
|
| 1525 |
-
if ($query[$i] != ' ')
|
| 1526 |
-
$spaceAllowed = false;
|
| 1527 |
-
$tmp .= $query[$i];
|
| 1528 |
-
$i++;
|
| 1529 |
-
}
|
| 1530 |
-
$return[] = $tmp;
|
| 1531 |
-
// ATTRS
|
| 1532 |
-
} else if ($c == '[') {
|
| 1533 |
-
$stack = 1;
|
| 1534 |
-
$tmp .= $c;
|
| 1535 |
-
while( isset($query[++$i])) {
|
| 1536 |
-
$tmp .= $query[$i];
|
| 1537 |
-
if ( $query[$i] == '[') {
|
| 1538 |
-
$stack++;
|
| 1539 |
-
} else if ( $query[$i] == ']') {
|
| 1540 |
-
$stack--;
|
| 1541 |
-
if (! $stack )
|
| 1542 |
-
break;
|
| 1543 |
-
}
|
| 1544 |
-
}
|
| 1545 |
-
$return[] = $tmp;
|
| 1546 |
-
$i++;
|
| 1547 |
-
// PSEUDO CLASSES
|
| 1548 |
-
} else if ($c == ':') {
|
| 1549 |
-
$stack = 1;
|
| 1550 |
-
$tmp .= $query[$i++];
|
| 1551 |
-
while( isset($query[$i]) && ($this->isChar($query[$i]) || in_array($query[$i], $pseudoChars))) {
|
| 1552 |
-
$tmp .= $query[$i];
|
| 1553 |
-
$i++;
|
| 1554 |
-
}
|
| 1555 |
-
// with arguments ?
|
| 1556 |
-
if ( isset($query[$i]) && $query[$i] == '(') {
|
| 1557 |
-
$tmp .= $query[$i];
|
| 1558 |
-
$stack = 1;
|
| 1559 |
-
while( isset($query[++$i])) {
|
| 1560 |
-
$tmp .= $query[$i];
|
| 1561 |
-
if ( $query[$i] == '(') {
|
| 1562 |
-
$stack++;
|
| 1563 |
-
} else if ( $query[$i] == ')') {
|
| 1564 |
-
$stack--;
|
| 1565 |
-
if (! $stack )
|
| 1566 |
-
break;
|
| 1567 |
-
}
|
| 1568 |
-
}
|
| 1569 |
-
$return[] = $tmp;
|
| 1570 |
-
$i++;
|
| 1571 |
-
} else {
|
| 1572 |
-
$return[] = $tmp;
|
| 1573 |
-
}
|
| 1574 |
-
} else {
|
| 1575 |
-
$i++;
|
| 1576 |
-
}
|
| 1577 |
-
}
|
| 1578 |
-
foreach($queries as $k => $q) {
|
| 1579 |
-
if (isset($q[0])) {
|
| 1580 |
-
if (isset($q[0][0]) && $q[0][0] == ':')
|
| 1581 |
-
array_unshift($queries[$k], '*');
|
| 1582 |
-
if ($q[0] != '>')
|
| 1583 |
-
array_unshift($queries[$k], ' ');
|
| 1584 |
-
}
|
| 1585 |
-
}
|
| 1586 |
-
return $queries;
|
| 1587 |
-
}
|
| 1588 |
-
/**
|
| 1589 |
-
* Return matched DOM nodes.
|
| 1590 |
-
*
|
| 1591 |
-
* @param int $index
|
| 1592 |
-
* @return array|DOMElement Single DOMElement or array of DOMElement.
|
| 1593 |
-
*/
|
| 1594 |
-
public function get($index = null, $callback1 = null, $callback2 = null, $callback3 = null) {
|
| 1595 |
-
$return = isset($index)
|
| 1596 |
-
? (isset($this->elements[$index]) ? $this->elements[$index] : null)
|
| 1597 |
-
: $this->elements;
|
| 1598 |
-
// pass thou callbacks
|
| 1599 |
-
$args = func_get_args();
|
| 1600 |
-
$args = array_slice($args, 1);
|
| 1601 |
-
foreach($args as $callback) {
|
| 1602 |
-
if (is_array($return))
|
| 1603 |
-
foreach($return as $k => $v)
|
| 1604 |
-
$return[$k] = phpQuery::callbackRun($callback, array($v));
|
| 1605 |
-
else
|
| 1606 |
-
$return = phpQuery::callbackRun($callback, array($return));
|
| 1607 |
-
}
|
| 1608 |
-
return $return;
|
| 1609 |
-
}
|
| 1610 |
-
/**
|
| 1611 |
-
* Return matched DOM nodes.
|
| 1612 |
-
* jQuery difference.
|
| 1613 |
-
*
|
| 1614 |
-
* @param int $index
|
| 1615 |
-
* @return array|string Returns string if $index != null
|
| 1616 |
-
* @todo implement callbacks
|
| 1617 |
-
* @todo return only arrays ?
|
| 1618 |
-
* @todo maybe other name...
|
| 1619 |
-
*/
|
| 1620 |
-
public function getString($index = null, $callback1 = null, $callback2 = null, $callback3 = null) {
|
| 1621 |
-
if ($index)
|
| 1622 |
-
$return = $this->eq($index)->text();
|
| 1623 |
-
else {
|
| 1624 |
-
$return = array();
|
| 1625 |
-
for($i = 0; $i < $this->size(); $i++) {
|
| 1626 |
-
$return[] = $this->eq($i)->text();
|
| 1627 |
-
}
|
| 1628 |
-
}
|
| 1629 |
-
// pass thou callbacks
|
| 1630 |
-
$args = func_get_args();
|
| 1631 |
-
$args = array_slice($args, 1);
|
| 1632 |
-
foreach($args as $callback) {
|
| 1633 |
-
$return = phpQuery::callbackRun($callback, array($return));
|
| 1634 |
-
}
|
| 1635 |
-
return $return;
|
| 1636 |
-
}
|
| 1637 |
-
/**
|
| 1638 |
-
* Return matched DOM nodes.
|
| 1639 |
-
* jQuery difference.
|
| 1640 |
-
*
|
| 1641 |
-
* @param int $index
|
| 1642 |
-
* @return array|string Returns string if $index != null
|
| 1643 |
-
* @todo implement callbacks
|
| 1644 |
-
* @todo return only arrays ?
|
| 1645 |
-
* @todo maybe other name...
|
| 1646 |
-
*/
|
| 1647 |
-
public function getStrings($index = null, $callback1 = null, $callback2 = null, $callback3 = null) {
|
| 1648 |
-
if ($index)
|
| 1649 |
-
$return = $this->eq($index)->text();
|
| 1650 |
-
else {
|
| 1651 |
-
$return = array();
|
| 1652 |
-
for($i = 0; $i < $this->size(); $i++) {
|
| 1653 |
-
$return[] = $this->eq($i)->text();
|
| 1654 |
-
}
|
| 1655 |
-
// pass thou callbacks
|
| 1656 |
-
$args = func_get_args();
|
| 1657 |
-
$args = array_slice($args, 1);
|
| 1658 |
-
}
|
| 1659 |
-
foreach($args as $callback) {
|
| 1660 |
-
if (is_array($return))
|
| 1661 |
-
foreach($return as $k => $v)
|
| 1662 |
-
$return[$k] = phpQuery::callbackRun($callback, array($v));
|
| 1663 |
-
else
|
| 1664 |
-
$return = phpQuery::callbackRun($callback, array($return));
|
| 1665 |
-
}
|
| 1666 |
-
return $return;
|
| 1667 |
-
}
|
| 1668 |
-
/**
|
| 1669 |
-
* Returns new instance of actual class.
|
| 1670 |
-
*
|
| 1671 |
-
* @param array $newStack Optional. Will replace old stack with new and move old one to history.c
|
| 1672 |
-
*/
|
| 1673 |
-
public function newInstance($newStack = null) {
|
| 1674 |
-
$class = get_class($this);
|
| 1675 |
-
// support inheritance by passing old object to overloaded constructor
|
| 1676 |
-
$new = $class != 'phpQuery'
|
| 1677 |
-
? new $class($this, $this->getDocumentID())
|
| 1678 |
-
: new phpQueryObject($this->getDocumentID());
|
| 1679 |
-
$new->previous = $this;
|
| 1680 |
-
if (is_null($newStack)) {
|
| 1681 |
-
$new->elements = $this->elements;
|
| 1682 |
-
if ($this->elementsBackup)
|
| 1683 |
-
$this->elements = $this->elementsBackup;
|
| 1684 |
-
} else if (is_string($newStack)) {
|
| 1685 |
-
$new->elements = phpQuery::pq($newStack, $this->getDocumentID())->stack();
|
| 1686 |
-
} else {
|
| 1687 |
-
$new->elements = $newStack;
|
| 1688 |
-
}
|
| 1689 |
-
return $new;
|
| 1690 |
-
}
|
| 1691 |
-
/**
|
| 1692 |
-
* Enter description here...
|
| 1693 |
-
*
|
| 1694 |
-
* In the future, when PHP will support XLS 2.0, then we would do that this way:
|
| 1695 |
-
* contains(tokenize(@class, '\s'), "something")
|
| 1696 |
-
* @param unknown_type $class
|
| 1697 |
-
* @param unknown_type $node
|
| 1698 |
-
* @return boolean
|
| 1699 |
-
* @access private
|
| 1700 |
-
*/
|
| 1701 |
-
protected function matchClasses($class, $node) {
|
| 1702 |
-
// multi-class
|
| 1703 |
-
if ( mb_strpos($class, '.', 1)) {
|
| 1704 |
-
$classes = explode('.', substr($class, 1));
|
| 1705 |
-
$classesCount = count( $classes );
|
| 1706 |
-
$nodeClasses = explode(' ', $node->getAttribute('class') );
|
| 1707 |
-
$nodeClassesCount = count( $nodeClasses );
|
| 1708 |
-
if ( $classesCount > $nodeClassesCount )
|
| 1709 |
-
return false;
|
| 1710 |
-
$diff = count(
|
| 1711 |
-
array_diff(
|
| 1712 |
-
$classes,
|
| 1713 |
-
$nodeClasses
|
| 1714 |
-
)
|
| 1715 |
-
);
|
| 1716 |
-
if (! $diff )
|
| 1717 |
-
return true;
|
| 1718 |
-
// single-class
|
| 1719 |
-
} else {
|
| 1720 |
-
return in_array(
|
| 1721 |
-
// strip leading dot from class name
|
| 1722 |
-
substr($class, 1),
|
| 1723 |
-
// get classes for element as array
|
| 1724 |
-
explode(' ', $node->getAttribute('class') )
|
| 1725 |
-
);
|
| 1726 |
-
}
|
| 1727 |
-
}
|
| 1728 |
-
/**
|
| 1729 |
-
* @access private
|
| 1730 |
-
*/
|
| 1731 |
-
protected function runQuery($XQuery, $selector = null, $compare = null) {
|
| 1732 |
-
if ($compare && ! method_exists($this, $compare))
|
| 1733 |
-
return false;
|
| 1734 |
-
$stack = array();
|
| 1735 |
-
if (! $this->elements)
|
| 1736 |
-
$this->debug('Stack empty, skipping...');
|
| 1737 |
-
// var_dump($this->elements[0]->nodeType);
|
| 1738 |
-
// element, document
|
| 1739 |
-
foreach($this->stack(array(1, 9, 13)) as $k => $stackNode) {
|
| 1740 |
-
$detachAfter = false;
|
| 1741 |
-
// to work on detached nodes we need temporary place them somewhere
|
| 1742 |
-
// thats because context xpath queries sucks ;]
|
| 1743 |
-
$testNode = $stackNode;
|
| 1744 |
-
while ($testNode) {
|
| 1745 |
-
if (! $testNode->parentNode && ! $this->isRoot($testNode)) {
|
| 1746 |
-
$this->root->appendChild($testNode);
|
| 1747 |
-
$detachAfter = $testNode;
|
| 1748 |
-
break;
|
| 1749 |
-
}
|
| 1750 |
-
$testNode = isset($testNode->parentNode)
|
| 1751 |
-
? $testNode->parentNode
|
| 1752 |
-
: null;
|
| 1753 |
-
}
|
| 1754 |
-
// XXX tmp ?
|
| 1755 |
-
$xpath = $this->documentWrapper->isXHTML
|
| 1756 |
-
? $this->getNodeXpath($stackNode, 'html')
|
| 1757 |
-
: $this->getNodeXpath($stackNode);
|
| 1758 |
-
// FIXME pseudoclasses-only query, support XML
|
| 1759 |
-
$query = $XQuery == '//' && $xpath == '/html[1]'
|
| 1760 |
-
? '//*'
|
| 1761 |
-
: $xpath.$XQuery;
|
| 1762 |
-
$this->debug("XPATH: {$query}");
|
| 1763 |
-
// run query, get elements
|
| 1764 |
-
$nodes = $this->xpath->query($query);
|
| 1765 |
-
$this->debug("QUERY FETCHED");
|
| 1766 |
-
if (! $nodes->length )
|
| 1767 |
-
$this->debug('Nothing found');
|
| 1768 |
-
$debug = array();
|
| 1769 |
-
foreach($nodes as $node) {
|
| 1770 |
-
$matched = false;
|
| 1771 |
-
if ( $compare) {
|
| 1772 |
-
phpQuery::$debug ?
|
| 1773 |
-
$this->debug("Found: ".$this->whois( $node ).", comparing with {$compare}()")
|
| 1774 |
-
: null;
|
| 1775 |
-
$phpQueryDebug = phpQuery::$debug;
|
| 1776 |
-
phpQuery::$debug = false;
|
| 1777 |
-
// TODO ??? use phpQuery::callbackRun()
|
| 1778 |
-
if (call_user_func_array(array($this, $compare), array($selector, $node)))
|
| 1779 |
-
$matched = true;
|
| 1780 |
-
phpQuery::$debug = $phpQueryDebug;
|
| 1781 |
-
} else {
|
| 1782 |
-
$matched = true;
|
| 1783 |
-
}
|
| 1784 |
-
if ( $matched) {
|
| 1785 |
-
if (phpQuery::$debug)
|
| 1786 |
-
$debug[] = $this->whois( $node );
|
| 1787 |
-
$stack[] = $node;
|
| 1788 |
-
}
|
| 1789 |
-
}
|
| 1790 |
-
if (phpQuery::$debug) {
|
| 1791 |
-
$this->debug("Matched ".count($debug).": ".implode(', ', $debug));
|
| 1792 |
-
}
|
| 1793 |
-
if ($detachAfter)
|
| 1794 |
-
$this->root->removeChild($detachAfter);
|
| 1795 |
-
}
|
| 1796 |
-
$this->elements = $stack;
|
| 1797 |
-
}
|
| 1798 |
-
/**
|
| 1799 |
-
* Enter description here...
|
| 1800 |
-
*
|
| 1801 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 1802 |
-
*/
|
| 1803 |
-
public function find($selectors, $context = null, $noHistory = false) {
|
| 1804 |
-
if (!$noHistory)
|
| 1805 |
-
// backup last stack /for end()/
|
| 1806 |
-
$this->elementsBackup = $this->elements;
|
| 1807 |
-
// allow to define context
|
| 1808 |
-
// TODO combine code below with phpQuery::pq() context guessing code
|
| 1809 |
-
// as generic function
|
| 1810 |
-
if ($context) {
|
| 1811 |
-
if (! is_array($context) && $context instanceof DOMELEMENT)
|
| 1812 |
-
$this->elements = array($context);
|
| 1813 |
-
else if (is_array($context)) {
|
| 1814 |
-
$this->elements = array();
|
| 1815 |
-
foreach ($context as $c)
|
| 1816 |
-
if ($c instanceof DOMELEMENT)
|
| 1817 |
-
$this->elements[] = $c;
|
| 1818 |
-
} else if ( $context instanceof self )
|
| 1819 |
-
$this->elements = $context->elements;
|
| 1820 |
-
}
|
| 1821 |
-
$queries = $this->parseSelector($selectors);
|
| 1822 |
-
$this->debug(array('FIND', $selectors, $queries));
|
| 1823 |
-
$XQuery = '';
|
| 1824 |
-
// remember stack state because of multi-queries
|
| 1825 |
-
$oldStack = $this->elements;
|
| 1826 |
-
// here we will be keeping found elements
|
| 1827 |
-
$stack = array();
|
| 1828 |
-
foreach($queries as $selector) {
|
| 1829 |
-
$this->elements = $oldStack;
|
| 1830 |
-
$delimiterBefore = false;
|
| 1831 |
-
foreach($selector as $s) {
|
| 1832 |
-
// TAG
|
| 1833 |
-
$isTag = extension_loaded('mbstring') && phpQuery::$mbstringSupport
|
| 1834 |
-
? mb_ereg_match('^[\w|\||-]+$', $s) || $s == '*'
|
| 1835 |
-
: preg_match('@^[\w|\||-]+$@', $s) || $s == '*';
|
| 1836 |
-
if ($isTag) {
|
| 1837 |
-
if ($this->isXML()) {
|
| 1838 |
-
// namespace support
|
| 1839 |
-
if (mb_strpos($s, '|') !== false) {
|
| 1840 |
-
$ns = $tag = null;
|
| 1841 |
-
list($ns, $tag) = explode('|', $s);
|
| 1842 |
-
$XQuery .= "$ns:$tag";
|
| 1843 |
-
} else if ($s == '*') {
|
| 1844 |
-
$XQuery .= "*";
|
| 1845 |
-
} else {
|
| 1846 |
-
$XQuery .= "*[local-name()='$s']";
|
| 1847 |
-
}
|
| 1848 |
-
} else {
|
| 1849 |
-
$XQuery .= $s;
|
| 1850 |
-
}
|
| 1851 |
-
// ID
|
| 1852 |
-
} else if ($s[0] == '#') {
|
| 1853 |
-
if ($delimiterBefore)
|
| 1854 |
-
$XQuery .= '*';
|
| 1855 |
-
$XQuery .= "[@id='".substr($s, 1)."']";
|
| 1856 |
-
// ATTRIBUTES
|
| 1857 |
-
} else if ($s[0] == '[') {
|
| 1858 |
-
if ($delimiterBefore)
|
| 1859 |
-
$XQuery .= '*';
|
| 1860 |
-
// strip side brackets
|
| 1861 |
-
$attr = trim($s, '][');
|
| 1862 |
-
$execute = false;
|
| 1863 |
-
// attr with specifed value
|
| 1864 |
-
if (mb_strpos($s, '=')) {
|
| 1865 |
-
$value = null;
|
| 1866 |
-
list($attr, $value) = explode('=', $attr);
|
| 1867 |
-
$value = trim($value, "'\"");
|
| 1868 |
-
if ($this->isRegexp($attr)) {
|
| 1869 |
-
// cut regexp character
|
| 1870 |
-
$attr = substr($attr, 0, -1);
|
| 1871 |
-
$execute = true;
|
| 1872 |
-
$XQuery .= "[@{$attr}]";
|
| 1873 |
-
} else {
|
| 1874 |
-
$XQuery .= "[@{$attr}='{$value}']";
|
| 1875 |
-
}
|
| 1876 |
-
// attr without specified value
|
| 1877 |
-
} else {
|
| 1878 |
-
$XQuery .= "[@{$attr}]";
|
| 1879 |
-
}
|
| 1880 |
-
if ($execute) {
|
| 1881 |
-
$this->runQuery($XQuery, $s, 'is');
|
| 1882 |
-
$XQuery = '';
|
| 1883 |
-
if (! $this->length())
|
| 1884 |
-
break;
|
| 1885 |
-
}
|
| 1886 |
-
// CLASSES
|
| 1887 |
-
} else if ($s[0] == '.') {
|
| 1888 |
-
// TODO use return $this->find("./self::*[contains(concat(\" \",@class,\" \"), \" $class \")]");
|
| 1889 |
-
// thx wizDom ;)
|
| 1890 |
-
if ($delimiterBefore)
|
| 1891 |
-
$XQuery .= '*';
|
| 1892 |
-
$XQuery .= '[@class]';
|
| 1893 |
-
$this->runQuery($XQuery, $s, 'matchClasses');
|
| 1894 |
-
$XQuery = '';
|
| 1895 |
-
if (! $this->length() )
|
| 1896 |
-
break;
|
| 1897 |
-
// ~ General Sibling Selector
|
| 1898 |
-
} else if ($s[0] == '~') {
|
| 1899 |
-
$this->runQuery($XQuery);
|
| 1900 |
-
$XQuery = '';
|
| 1901 |
-
$this->elements = $this
|
| 1902 |
-
->siblings(
|
| 1903 |
-
substr($s, 1)
|
| 1904 |
-
)->elements;
|
| 1905 |
-
if (! $this->length() )
|
| 1906 |
-
break;
|
| 1907 |
-
// + Adjacent sibling selectors
|
| 1908 |
-
} else if ($s[0] == '+') {
|
| 1909 |
-
// TODO /following-sibling::
|
| 1910 |
-
$this->runQuery($XQuery);
|
| 1911 |
-
$XQuery = '';
|
| 1912 |
-
$subSelector = substr($s, 1);
|
| 1913 |
-
$subElements = $this->elements;
|
| 1914 |
-
$this->elements = array();
|
| 1915 |
-
foreach($subElements as $node) {
|
| 1916 |
-
// search first DOMElement sibling
|
| 1917 |
-
$test = $node->nextSibling;
|
| 1918 |
-
while($test && ! ($test instanceof DOMELEMENT))
|
| 1919 |
-
$test = $test->nextSibling;
|
| 1920 |
-
if ($test && $this->is($subSelector, $test))
|
| 1921 |
-
$this->elements[] = $test;
|
| 1922 |
-
}
|
| 1923 |
-
if (! $this->length() )
|
| 1924 |
-
break;
|
| 1925 |
-
// PSEUDO CLASSES
|
| 1926 |
-
} else if ($s[0] == ':') {
|
| 1927 |
-
// TODO optimization for :first :last
|
| 1928 |
-
if ($XQuery) {
|
| 1929 |
-
$this->runQuery($XQuery);
|
| 1930 |
-
$XQuery = '';
|
| 1931 |
-
}
|
| 1932 |
-
if (! $this->length())
|
| 1933 |
-
break;
|
| 1934 |
-
$this->pseudoClasses($s);
|
| 1935 |
-
if (! $this->length())
|
| 1936 |
-
break;
|
| 1937 |
-
// DIRECT DESCENDANDS
|
| 1938 |
-
} else if ($s == '>') {
|
| 1939 |
-
$XQuery .= '/';
|
| 1940 |
-
$delimiterBefore = 2;
|
| 1941 |
-
// ALL DESCENDANDS
|
| 1942 |
-
} else if ($s == ' ') {
|
| 1943 |
-
$XQuery .= '//';
|
| 1944 |
-
$delimiterBefore = 2;
|
| 1945 |
-
// ERRORS
|
| 1946 |
-
} else {
|
| 1947 |
-
phpQuery::debug("Unrecognized token '$s'");
|
| 1948 |
-
}
|
| 1949 |
-
$delimiterBefore = $delimiterBefore === 2;
|
| 1950 |
-
}
|
| 1951 |
-
// run query if any
|
| 1952 |
-
if ($XQuery && $XQuery != '//') {
|
| 1953 |
-
$this->runQuery($XQuery);
|
| 1954 |
-
$XQuery = '';
|
| 1955 |
-
}
|
| 1956 |
-
foreach($this->elements as $node)
|
| 1957 |
-
if (! $this->elementsContainsNode($node, $stack))
|
| 1958 |
-
$stack[] = $node;
|
| 1959 |
-
}
|
| 1960 |
-
$this->elements = $stack;
|
| 1961 |
-
return $this->newInstance();
|
| 1962 |
-
}
|
| 1963 |
-
/**
|
| 1964 |
-
* @todo create API for classes with pseudoselectors
|
| 1965 |
-
* @access private
|
| 1966 |
-
*/
|
| 1967 |
-
protected function pseudoClasses($class) {
|
| 1968 |
-
// TODO clean args parsing ?
|
| 1969 |
-
$class = ltrim($class, ':');
|
| 1970 |
-
$haveArgs = mb_strpos($class, '(');
|
| 1971 |
-
if ($haveArgs !== false) {
|
| 1972 |
-
$args = substr($class, $haveArgs+1, -1);
|
| 1973 |
-
$class = substr($class, 0, $haveArgs);
|
| 1974 |
-
}
|
| 1975 |
-
switch($class) {
|
| 1976 |
-
case 'even':
|
| 1977 |
-
case 'odd':
|
| 1978 |
-
$stack = array();
|
| 1979 |
-
foreach($this->elements as $i => $node) {
|
| 1980 |
-
if ($class == 'even' && ($i%2) == 0)
|
| 1981 |
-
$stack[] = $node;
|
| 1982 |
-
else if ( $class == 'odd' && $i % 2 )
|
| 1983 |
-
$stack[] = $node;
|
| 1984 |
-
}
|
| 1985 |
-
$this->elements = $stack;
|
| 1986 |
-
break;
|
| 1987 |
-
case 'eq':
|
| 1988 |
-
$k = intval($args);
|
| 1989 |
-
$this->elements = isset( $this->elements[$k] )
|
| 1990 |
-
? array( $this->elements[$k] )
|
| 1991 |
-
: array();
|
| 1992 |
-
break;
|
| 1993 |
-
case 'gt':
|
| 1994 |
-
$this->elements = array_slice($this->elements, $args+1);
|
| 1995 |
-
break;
|
| 1996 |
-
case 'lt':
|
| 1997 |
-
$this->elements = array_slice($this->elements, 0, $args+1);
|
| 1998 |
-
break;
|
| 1999 |
-
case 'first':
|
| 2000 |
-
if (isset($this->elements[0]))
|
| 2001 |
-
$this->elements = array($this->elements[0]);
|
| 2002 |
-
break;
|
| 2003 |
-
case 'last':
|
| 2004 |
-
if ($this->elements)
|
| 2005 |
-
$this->elements = array($this->elements[count($this->elements)-1]);
|
| 2006 |
-
break;
|
| 2007 |
-
/*case 'parent':
|
| 2008 |
-
$stack = array();
|
| 2009 |
-
foreach($this->elements as $node) {
|
| 2010 |
-
if ( $node->childNodes->length )
|
| 2011 |
-
$stack[] = $node;
|
| 2012 |
-
}
|
| 2013 |
-
$this->elements = $stack;
|
| 2014 |
-
break;*/
|
| 2015 |
-
case 'contains':
|
| 2016 |
-
$text = trim($args, "\"'");
|
| 2017 |
-
$stack = array();
|
| 2018 |
-
foreach($this->elements as $node) {
|
| 2019 |
-
if (mb_stripos($node->textContent, $text) === false)
|
| 2020 |
-
continue;
|
| 2021 |
-
$stack[] = $node;
|
| 2022 |
-
}
|
| 2023 |
-
$this->elements = $stack;
|
| 2024 |
-
break;
|
| 2025 |
-
case 'not':
|
| 2026 |
-
$selector = self::unQuote($args);
|
| 2027 |
-
$this->elements = $this->not($selector)->stack();
|
| 2028 |
-
break;
|
| 2029 |
-
case 'slice':
|
| 2030 |
-
// TODO jQuery difference ?
|
| 2031 |
-
$args = explode(',',
|
| 2032 |
-
str_replace(', ', ',', trim($args, "\"'"))
|
| 2033 |
-
);
|
| 2034 |
-
$start = $args[0];
|
| 2035 |
-
$end = isset($args[1])
|
| 2036 |
-
? $args[1]
|
| 2037 |
-
: null;
|
| 2038 |
-
if ($end > 0)
|
| 2039 |
-
$end = $end-$start;
|
| 2040 |
-
$this->elements = array_slice($this->elements, $start, $end);
|
| 2041 |
-
break;
|
| 2042 |
-
case 'has':
|
| 2043 |
-
$selector = trim($args, "\"'");
|
| 2044 |
-
$stack = array();
|
| 2045 |
-
foreach($this->stack(1) as $el) {
|
| 2046 |
-
if ($this->find($selector, $el, true)->length)
|
| 2047 |
-
$stack[] = $el;
|
| 2048 |
-
}
|
| 2049 |
-
$this->elements = $stack;
|
| 2050 |
-
break;
|
| 2051 |
-
case 'submit':
|
| 2052 |
-
case 'reset':
|
| 2053 |
-
$this->elements = phpQuery::merge(
|
| 2054 |
-
$this->map(array($this, 'is'),
|
| 2055 |
-
"input[type=$class]", new CallbackParam()
|
| 2056 |
-
),
|
| 2057 |
-
$this->map(array($this, 'is'),
|
| 2058 |
-
"button[type=$class]", new CallbackParam()
|
| 2059 |
-
)
|
| 2060 |
-
);
|
| 2061 |
-
break;
|
| 2062 |
-
// $stack = array();
|
| 2063 |
-
// foreach($this->elements as $node)
|
| 2064 |
-
// if ($node->is('input[type=submit]') || $node->is('button[type=submit]'))
|
| 2065 |
-
// $stack[] = $el;
|
| 2066 |
-
// $this->elements = $stack;
|
| 2067 |
-
case 'input':
|
| 2068 |
-
$this->elements = $this->map(
|
| 2069 |
-
array($this, 'is'),
|
| 2070 |
-
'input', new CallbackParam()
|
| 2071 |
-
)->elements;
|
| 2072 |
-
break;
|
| 2073 |
-
case 'password':
|
| 2074 |
-
case 'checkbox':
|
| 2075 |
-
case 'radio':
|
| 2076 |
-
case 'hidden':
|
| 2077 |
-
case 'image':
|
| 2078 |
-
case 'file':
|
| 2079 |
-
$this->elements = $this->map(
|
| 2080 |
-
array($this, 'is'),
|
| 2081 |
-
"input[type=$class]", new CallbackParam()
|
| 2082 |
-
)->elements;
|
| 2083 |
-
break;
|
| 2084 |
-
case 'parent':
|
| 2085 |
-
$this->elements = $this->map(
|
| 2086 |
-
create_function('$node', '
|
| 2087 |
-
return $node instanceof DOMELEMENT && $node->childNodes->length
|
| 2088 |
-
? $node : null;')
|
| 2089 |
-
)->elements;
|
| 2090 |
-
break;
|
| 2091 |
-
case 'empty':
|
| 2092 |
-
$this->elements = $this->map(
|
| 2093 |
-
create_function('$node', '
|
| 2094 |
-
return $node instanceof DOMELEMENT && $node->childNodes->length
|
| 2095 |
-
? null : $node;')
|
| 2096 |
-
)->elements;
|
| 2097 |
-
break;
|
| 2098 |
-
case 'disabled':
|
| 2099 |
-
case 'selected':
|
| 2100 |
-
case 'checked':
|
| 2101 |
-
$this->elements = $this->map(
|
| 2102 |
-
array($this, 'is'),
|
| 2103 |
-
"[$class]", new CallbackParam()
|
| 2104 |
-
)->elements;
|
| 2105 |
-
break;
|
| 2106 |
-
case 'enabled':
|
| 2107 |
-
$this->elements = $this->map(
|
| 2108 |
-
create_function('$node', '
|
| 2109 |
-
return pq($node)->not(":disabled") ? $node : null;')
|
| 2110 |
-
)->elements;
|
| 2111 |
-
break;
|
| 2112 |
-
case 'header':
|
| 2113 |
-
$this->elements = $this->map(
|
| 2114 |
-
create_function('$node',
|
| 2115 |
-
'$isHeader = isset($node->tagName) && in_array($node->tagName, array(
|
| 2116 |
-
"h1", "h2", "h3", "h4", "h5", "h6", "h7"
|
| 2117 |
-
));
|
| 2118 |
-
return $isHeader
|
| 2119 |
-
? $node
|
| 2120 |
-
: null;')
|
| 2121 |
-
)->elements;
|
| 2122 |
-
// $this->elements = $this->map(
|
| 2123 |
-
// create_function('$node', '$node = pq($node);
|
| 2124 |
-
// return $node->is("h1")
|
| 2125 |
-
// || $node->is("h2")
|
| 2126 |
-
// || $node->is("h3")
|
| 2127 |
-
// || $node->is("h4")
|
| 2128 |
-
// || $node->is("h5")
|
| 2129 |
-
// || $node->is("h6")
|
| 2130 |
-
// || $node->is("h7")
|
| 2131 |
-
// ? $node
|
| 2132 |
-
// : null;')
|
| 2133 |
-
// )->elements;
|
| 2134 |
-
break;
|
| 2135 |
-
case 'only-child':
|
| 2136 |
-
$this->elements = $this->map(
|
| 2137 |
-
create_function('$node',
|
| 2138 |
-
'return pq($node)->siblings()->size() == 0 ? $node : null;')
|
| 2139 |
-
)->elements;
|
| 2140 |
-
break;
|
| 2141 |
-
case 'first-child':
|
| 2142 |
-
$this->elements = $this->map(
|
| 2143 |
-
create_function('$node', 'return pq($node)->prevAll()->size() == 0 ? $node : null;')
|
| 2144 |
-
)->elements;
|
| 2145 |
-
break;
|
| 2146 |
-
case 'last-child':
|
| 2147 |
-
$this->elements = $this->map(
|
| 2148 |
-
create_function('$node', 'return pq($node)->nextAll()->size() == 0 ? $node : null;')
|
| 2149 |
-
)->elements;
|
| 2150 |
-
break;
|
| 2151 |
-
case 'nth-child':
|
| 2152 |
-
$param = trim($args, "\"'");
|
| 2153 |
-
if (! $param)
|
| 2154 |
-
break;
|
| 2155 |
-
// nth-child(n+b) to nth-child(1n+b)
|
| 2156 |
-
if ($param{0} == 'n')
|
| 2157 |
-
$param = '1'.$param;
|
| 2158 |
-
// :nth-child(index/even/odd/equation)
|
| 2159 |
-
if ($param == 'even' || $param == 'odd')
|
| 2160 |
-
$mapped = $this->map(
|
| 2161 |
-
create_function('$node, $param',
|
| 2162 |
-
'$index = pq($node)->prevAll()->size()+1;
|
| 2163 |
-
if ($param == "even" && ($index%2) == 0)
|
| 2164 |
-
return $node;
|
| 2165 |
-
else if ($param == "odd" && $index%2 == 1)
|
| 2166 |
-
return $node;
|
| 2167 |
-
else
|
| 2168 |
-
return null;'),
|
| 2169 |
-
new CallbackParam(), $param
|
| 2170 |
-
);
|
| 2171 |
-
else if (mb_strlen($param) > 1 && $param{1} == 'n')
|
| 2172 |
-
// an+b
|
| 2173 |
-
$mapped = $this->map(
|
| 2174 |
-
create_function('$node, $param',
|
| 2175 |
-
'$prevs = pq($node)->prevAll()->size();
|
| 2176 |
-
$index = 1+$prevs;
|
| 2177 |
-
$b = mb_strlen($param) > 3
|
| 2178 |
-
? $param{3}
|
| 2179 |
-
: 0;
|
| 2180 |
-
$a = $param{0};
|
| 2181 |
-
if ($b && $param{2} == "-")
|
| 2182 |
-
$b = -$b;
|
| 2183 |
-
if ($a > 0) {
|
| 2184 |
-
return ($index-$b)%$a == 0
|
| 2185 |
-
? $node
|
| 2186 |
-
: null;
|
| 2187 |
-
phpQuery::debug($a."*".floor($index/$a)."+$b-1 == ".($a*floor($index/$a)+$b-1)." ?= $prevs");
|
| 2188 |
-
return $a*floor($index/$a)+$b-1 == $prevs
|
| 2189 |
-
? $node
|
| 2190 |
-
: null;
|
| 2191 |
-
} else if ($a == 0)
|
| 2192 |
-
return $index == $b
|
| 2193 |
-
? $node
|
| 2194 |
-
: null;
|
| 2195 |
-
else
|
| 2196 |
-
// negative value
|
| 2197 |
-
return $index <= $b
|
| 2198 |
-
? $node
|
| 2199 |
-
: null;
|
| 2200 |
-
// if (! $b)
|
| 2201 |
-
// return $index%$a == 0
|
| 2202 |
-
// ? $node
|
| 2203 |
-
// : null;
|
| 2204 |
-
// else
|
| 2205 |
-
// return ($index-$b)%$a == 0
|
| 2206 |
-
// ? $node
|
| 2207 |
-
// : null;
|
| 2208 |
-
'),
|
| 2209 |
-
new CallbackParam(), $param
|
| 2210 |
-
);
|
| 2211 |
-
else
|
| 2212 |
-
// index
|
| 2213 |
-
$mapped = $this->map(
|
| 2214 |
-
create_function('$node, $index',
|
| 2215 |
-
'$prevs = pq($node)->prevAll()->size();
|
| 2216 |
-
if ($prevs && $prevs == $index-1)
|
| 2217 |
-
return $node;
|
| 2218 |
-
else if (! $prevs && $index == 1)
|
| 2219 |
-
return $node;
|
| 2220 |
-
else
|
| 2221 |
-
return null;'),
|
| 2222 |
-
new CallbackParam(), $param
|
| 2223 |
-
);
|
| 2224 |
-
$this->elements = $mapped->elements;
|
| 2225 |
-
break;
|
| 2226 |
-
default:
|
| 2227 |
-
$this->debug("Unknown pseudoclass '{$class}', skipping...");
|
| 2228 |
-
}
|
| 2229 |
-
}
|
| 2230 |
-
/**
|
| 2231 |
-
* @access private
|
| 2232 |
-
*/
|
| 2233 |
-
protected function __pseudoClassParam($paramsString) {
|
| 2234 |
-
// TODO;
|
| 2235 |
-
}
|
| 2236 |
-
/**
|
| 2237 |
-
* Enter description here...
|
| 2238 |
-
*
|
| 2239 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2240 |
-
*/
|
| 2241 |
-
public function is($selector, $nodes = null) {
|
| 2242 |
-
phpQuery::debug(array("Is:", $selector));
|
| 2243 |
-
if (! $selector)
|
| 2244 |
-
return false;
|
| 2245 |
-
$oldStack = $this->elements;
|
| 2246 |
-
$returnArray = false;
|
| 2247 |
-
if ($nodes && is_array($nodes)) {
|
| 2248 |
-
$this->elements = $nodes;
|
| 2249 |
-
} else if ($nodes)
|
| 2250 |
-
$this->elements = array($nodes);
|
| 2251 |
-
$this->filter($selector, true);
|
| 2252 |
-
$stack = $this->elements;
|
| 2253 |
-
$this->elements = $oldStack;
|
| 2254 |
-
if ($nodes)
|
| 2255 |
-
return $stack ? $stack : null;
|
| 2256 |
-
return (bool)count($stack);
|
| 2257 |
-
}
|
| 2258 |
-
/**
|
| 2259 |
-
* Enter description here...
|
| 2260 |
-
* jQuery difference.
|
| 2261 |
-
*
|
| 2262 |
-
* Callback:
|
| 2263 |
-
* - $index int
|
| 2264 |
-
* - $node DOMNode
|
| 2265 |
-
*
|
| 2266 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2267 |
-
* @link http://docs.jquery.com/Traversing/filter
|
| 2268 |
-
*/
|
| 2269 |
-
public function filterCallback($callback, $_skipHistory = false) {
|
| 2270 |
-
if (! $_skipHistory) {
|
| 2271 |
-
$this->elementsBackup = $this->elements;
|
| 2272 |
-
$this->debug("Filtering by callback");
|
| 2273 |
-
}
|
| 2274 |
-
$newStack = array();
|
| 2275 |
-
foreach($this->elements as $index => $node) {
|
| 2276 |
-
$result = phpQuery::callbackRun($callback, array($index, $node));
|
| 2277 |
-
if (is_null($result) || (! is_null($result) && $result))
|
| 2278 |
-
$newStack[] = $node;
|
| 2279 |
-
}
|
| 2280 |
-
$this->elements = $newStack;
|
| 2281 |
-
return $_skipHistory
|
| 2282 |
-
? $this
|
| 2283 |
-
: $this->newInstance();
|
| 2284 |
-
}
|
| 2285 |
-
/**
|
| 2286 |
-
* Enter description here...
|
| 2287 |
-
*
|
| 2288 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2289 |
-
* @link http://docs.jquery.com/Traversing/filter
|
| 2290 |
-
*/
|
| 2291 |
-
public function filter($selectors, $_skipHistory = false) {
|
| 2292 |
-
if ($selectors instanceof Callback OR $selectors instanceof Closure)
|
| 2293 |
-
return $this->filterCallback($selectors, $_skipHistory);
|
| 2294 |
-
if (! $_skipHistory)
|
| 2295 |
-
$this->elementsBackup = $this->elements;
|
| 2296 |
-
$notSimpleSelector = array(' ', '>', '~', '+', '/');
|
| 2297 |
-
if (! is_array($selectors))
|
| 2298 |
-
$selectors = $this->parseSelector($selectors);
|
| 2299 |
-
if (! $_skipHistory)
|
| 2300 |
-
$this->debug(array("Filtering:", $selectors));
|
| 2301 |
-
$finalStack = array();
|
| 2302 |
-
foreach($selectors as $selector) {
|
| 2303 |
-
$stack = array();
|
| 2304 |
-
if (! $selector)
|
| 2305 |
-
break;
|
| 2306 |
-
// avoid first space or /
|
| 2307 |
-
if (in_array($selector[0], $notSimpleSelector))
|
| 2308 |
-
$selector = array_slice($selector, 1);
|
| 2309 |
-
// PER NODE selector chunks
|
| 2310 |
-
foreach($this->stack() as $node) {
|
| 2311 |
-
$break = false;
|
| 2312 |
-
foreach($selector as $s) {
|
| 2313 |
-
if (!($node instanceof DOMELEMENT)) {
|
| 2314 |
-
// all besides DOMElement
|
| 2315 |
-
if ( $s[0] == '[') {
|
| 2316 |
-
$attr = trim($s, '[]');
|
| 2317 |
-
if ( mb_strpos($attr, '=')) {
|
| 2318 |
-
list( $attr, $val ) = explode('=', $attr);
|
| 2319 |
-
if ($attr == 'nodeType' && $node->nodeType != $val)
|
| 2320 |
-
$break = true;
|
| 2321 |
-
}
|
| 2322 |
-
} else
|
| 2323 |
-
$break = true;
|
| 2324 |
-
} else {
|
| 2325 |
-
// DOMElement only
|
| 2326 |
-
// ID
|
| 2327 |
-
if ( $s[0] == '#') {
|
| 2328 |
-
if ( $node->getAttribute('id') != substr($s, 1) )
|
| 2329 |
-
$break = true;
|
| 2330 |
-
// CLASSES
|
| 2331 |
-
} else if ( $s[0] == '.') {
|
| 2332 |
-
if (! $this->matchClasses( $s, $node ) )
|
| 2333 |
-
$break = true;
|
| 2334 |
-
// ATTRS
|
| 2335 |
-
} else if ( $s[0] == '[') {
|
| 2336 |
-
// strip side brackets
|
| 2337 |
-
$attr = trim($s, '[]');
|
| 2338 |
-
if (mb_strpos($attr, '=')) {
|
| 2339 |
-
list($attr, $val) = explode('=', $attr);
|
| 2340 |
-
$val = self::unQuote($val);
|
| 2341 |
-
if ($attr == 'nodeType') {
|
| 2342 |
-
if ($val != $node->nodeType)
|
| 2343 |
-
$break = true;
|
| 2344 |
-
} else if ($this->isRegexp($attr)) {
|
| 2345 |
-
$val = extension_loaded('mbstring') && phpQuery::$mbstringSupport
|
| 2346 |
-
? quotemeta(trim($val, '"\''))
|
| 2347 |
-
: preg_quote(trim($val, '"\''), '@');
|
| 2348 |
-
// switch last character
|
| 2349 |
-
switch( substr($attr, -1)) {
|
| 2350 |
-
// quotemeta used insted of preg_quote
|
| 2351 |
-
// http://code.google.com/p/phpquery/issues/detail?id=76
|
| 2352 |
-
case '^':
|
| 2353 |
-
$pattern = '^'.$val;
|
| 2354 |
-
break;
|
| 2355 |
-
case '*':
|
| 2356 |
-
$pattern = '.*'.$val.'.*';
|
| 2357 |
-
break;
|
| 2358 |
-
case '$':
|
| 2359 |
-
$pattern = '.*'.$val.'$';
|
| 2360 |
-
break;
|
| 2361 |
-
}
|
| 2362 |
-
// cut last character
|
| 2363 |
-
$attr = substr($attr, 0, -1);
|
| 2364 |
-
$isMatch = extension_loaded('mbstring') && phpQuery::$mbstringSupport
|
| 2365 |
-
? mb_ereg_match($pattern, $node->getAttribute($attr))
|
| 2366 |
-
: preg_match("@{$pattern}@", $node->getAttribute($attr));
|
| 2367 |
-
if (! $isMatch)
|
| 2368 |
-
$break = true;
|
| 2369 |
-
} else if ($node->getAttribute($attr) != $val)
|
| 2370 |
-
$break = true;
|
| 2371 |
-
} else if (! $node->hasAttribute($attr))
|
| 2372 |
-
$break = true;
|
| 2373 |
-
// PSEUDO CLASSES
|
| 2374 |
-
} else if ( $s[0] == ':') {
|
| 2375 |
-
// skip
|
| 2376 |
-
// TAG
|
| 2377 |
-
} else if (trim($s)) {
|
| 2378 |
-
if ($s != '*') {
|
| 2379 |
-
// TODO namespaces
|
| 2380 |
-
if (isset($node->tagName)) {
|
| 2381 |
-
if ($node->tagName != $s)
|
| 2382 |
-
$break = true;
|
| 2383 |
-
} else if ($s == 'html' && ! $this->isRoot($node))
|
| 2384 |
-
$break = true;
|
| 2385 |
-
}
|
| 2386 |
-
// AVOID NON-SIMPLE SELECTORS
|
| 2387 |
-
} else if (in_array($s, $notSimpleSelector)) {
|
| 2388 |
-
$break = true;
|
| 2389 |
-
$this->debug(array('Skipping non simple selector', $selector));
|
| 2390 |
-
}
|
| 2391 |
-
}
|
| 2392 |
-
if ($break)
|
| 2393 |
-
break;
|
| 2394 |
-
}
|
| 2395 |
-
// if element passed all chunks of selector - add it to new stack
|
| 2396 |
-
if (! $break )
|
| 2397 |
-
$stack[] = $node;
|
| 2398 |
-
}
|
| 2399 |
-
$tmpStack = $this->elements;
|
| 2400 |
-
$this->elements = $stack;
|
| 2401 |
-
// PER ALL NODES selector chunks
|
| 2402 |
-
foreach($selector as $s)
|
| 2403 |
-
// PSEUDO CLASSES
|
| 2404 |
-
if ($s[0] == ':')
|
| 2405 |
-
$this->pseudoClasses($s);
|
| 2406 |
-
foreach($this->elements as $node)
|
| 2407 |
-
// XXX it should be merged without duplicates
|
| 2408 |
-
// but jQuery doesnt do that
|
| 2409 |
-
$finalStack[] = $node;
|
| 2410 |
-
$this->elements = $tmpStack;
|
| 2411 |
-
}
|
| 2412 |
-
$this->elements = $finalStack;
|
| 2413 |
-
if ($_skipHistory) {
|
| 2414 |
-
return $this;
|
| 2415 |
-
} else {
|
| 2416 |
-
$this->debug("Stack length after filter(): ".count($finalStack));
|
| 2417 |
-
return $this->newInstance();
|
| 2418 |
-
}
|
| 2419 |
-
}
|
| 2420 |
-
/**
|
| 2421 |
-
*
|
| 2422 |
-
* @param $value
|
| 2423 |
-
* @return unknown_type
|
| 2424 |
-
* @TODO implement in all methods using passed parameters
|
| 2425 |
-
*/
|
| 2426 |
-
protected static function unQuote($value) {
|
| 2427 |
-
return $value[0] == '\'' || $value[0] == '"'
|
| 2428 |
-
? substr($value, 1, -1)
|
| 2429 |
-
: $value;
|
| 2430 |
-
}
|
| 2431 |
-
/**
|
| 2432 |
-
* Enter description here...
|
| 2433 |
-
*
|
| 2434 |
-
* @link http://docs.jquery.com/Ajax/load
|
| 2435 |
-
* @return phpQuery|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2436 |
-
* @todo Support $selector
|
| 2437 |
-
*/
|
| 2438 |
-
public function load($url, $data = null, $callback = null) {
|
| 2439 |
-
if ($data && ! is_array($data)) {
|
| 2440 |
-
$callback = $data;
|
| 2441 |
-
$data = null;
|
| 2442 |
-
}
|
| 2443 |
-
if (mb_strpos($url, ' ') !== false) {
|
| 2444 |
-
$matches = null;
|
| 2445 |
-
if (extension_loaded('mbstring') && phpQuery::$mbstringSupport)
|
| 2446 |
-
mb_ereg('^([^ ]+) (.*)$', $url, $matches);
|
| 2447 |
-
else
|
| 2448 |
-
preg_match('^([^ ]+) (.*)$', $url, $matches);
|
| 2449 |
-
$url = $matches[1];
|
| 2450 |
-
$selector = $matches[2];
|
| 2451 |
-
// FIXME this sucks, pass as callback param
|
| 2452 |
-
$this->_loadSelector = $selector;
|
| 2453 |
-
}
|
| 2454 |
-
$ajax = array(
|
| 2455 |
-
'url' => $url,
|
| 2456 |
-
'type' => $data ? 'POST' : 'GET',
|
| 2457 |
-
'data' => $data,
|
| 2458 |
-
'complete' => $callback,
|
| 2459 |
-
'success' => array($this, '__loadSuccess')
|
| 2460 |
-
);
|
| 2461 |
-
phpQuery::ajax($ajax);
|
| 2462 |
-
return $this;
|
| 2463 |
-
}
|
| 2464 |
-
/**
|
| 2465 |
-
* @access private
|
| 2466 |
-
* @param $html
|
| 2467 |
-
* @return unknown_type
|
| 2468 |
-
*/
|
| 2469 |
-
public function __loadSuccess($html) {
|
| 2470 |
-
if ($this->_loadSelector) {
|
| 2471 |
-
$html = phpQuery::newDocument($html)->find($this->_loadSelector);
|
| 2472 |
-
unset($this->_loadSelector);
|
| 2473 |
-
}
|
| 2474 |
-
foreach($this->stack(1) as $node) {
|
| 2475 |
-
phpQuery::pq($node, $this->getDocumentID())
|
| 2476 |
-
->markup($html);
|
| 2477 |
-
}
|
| 2478 |
-
}
|
| 2479 |
-
/**
|
| 2480 |
-
* Enter description here...
|
| 2481 |
-
*
|
| 2482 |
-
* @return phpQuery|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2483 |
-
* @todo
|
| 2484 |
-
*/
|
| 2485 |
-
public function css() {
|
| 2486 |
-
// TODO
|
| 2487 |
-
return $this;
|
| 2488 |
-
}
|
| 2489 |
-
/**
|
| 2490 |
-
* @todo
|
| 2491 |
-
*
|
| 2492 |
-
*/
|
| 2493 |
-
public function show(){
|
| 2494 |
-
// TODO
|
| 2495 |
-
return $this;
|
| 2496 |
-
}
|
| 2497 |
-
/**
|
| 2498 |
-
* @todo
|
| 2499 |
-
*
|
| 2500 |
-
*/
|
| 2501 |
-
public function hide(){
|
| 2502 |
-
// TODO
|
| 2503 |
-
return $this;
|
| 2504 |
-
}
|
| 2505 |
-
/**
|
| 2506 |
-
* Trigger a type of event on every matched element.
|
| 2507 |
-
*
|
| 2508 |
-
* @param unknown_type $type
|
| 2509 |
-
* @param unknown_type $data
|
| 2510 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2511 |
-
* @TODO support more than event in $type (space-separated)
|
| 2512 |
-
*/
|
| 2513 |
-
public function trigger($type, $data = array()) {
|
| 2514 |
-
foreach($this->elements as $node)
|
| 2515 |
-
phpQueryEvents::trigger($this->getDocumentID(), $type, $data, $node);
|
| 2516 |
-
return $this;
|
| 2517 |
-
}
|
| 2518 |
-
/**
|
| 2519 |
-
* This particular method triggers all bound event handlers on an element (for a specific event type) WITHOUT executing the browsers default actions.
|
| 2520 |
-
*
|
| 2521 |
-
* @param unknown_type $type
|
| 2522 |
-
* @param unknown_type $data
|
| 2523 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2524 |
-
* @TODO
|
| 2525 |
-
*/
|
| 2526 |
-
public function triggerHandler($type, $data = array()) {
|
| 2527 |
-
// TODO;
|
| 2528 |
-
}
|
| 2529 |
-
/**
|
| 2530 |
-
* Binds a handler to one or more events (like click) for each matched element.
|
| 2531 |
-
* Can also bind custom events.
|
| 2532 |
-
*
|
| 2533 |
-
* @param unknown_type $type
|
| 2534 |
-
* @param unknown_type $data Optional
|
| 2535 |
-
* @param unknown_type $callback
|
| 2536 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2537 |
-
* @TODO support '!' (exclusive) events
|
| 2538 |
-
* @TODO support more than event in $type (space-separated)
|
| 2539 |
-
*/
|
| 2540 |
-
public function bind($type, $data, $callback = null) {
|
| 2541 |
-
// TODO check if $data is callable, not using is_callable
|
| 2542 |
-
if (! isset($callback)) {
|
| 2543 |
-
$callback = $data;
|
| 2544 |
-
$data = null;
|
| 2545 |
-
}
|
| 2546 |
-
foreach($this->elements as $node)
|
| 2547 |
-
phpQueryEvents::add($this->getDocumentID(), $node, $type, $data, $callback);
|
| 2548 |
-
return $this;
|
| 2549 |
-
}
|
| 2550 |
-
/**
|
| 2551 |
-
* Enter description here...
|
| 2552 |
-
*
|
| 2553 |
-
* @param unknown_type $type
|
| 2554 |
-
* @param unknown_type $callback
|
| 2555 |
-
* @return unknown
|
| 2556 |
-
* @TODO namespace events
|
| 2557 |
-
* @TODO support more than event in $type (space-separated)
|
| 2558 |
-
*/
|
| 2559 |
-
public function unbind($type = null, $callback = null) {
|
| 2560 |
-
foreach($this->elements as $node)
|
| 2561 |
-
phpQueryEvents::remove($this->getDocumentID(), $node, $type, $callback);
|
| 2562 |
-
return $this;
|
| 2563 |
-
}
|
| 2564 |
-
/**
|
| 2565 |
-
* Enter description here...
|
| 2566 |
-
*
|
| 2567 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2568 |
-
*/
|
| 2569 |
-
public function change($callback = null) {
|
| 2570 |
-
if ($callback)
|
| 2571 |
-
return $this->bind('change', $callback);
|
| 2572 |
-
return $this->trigger('change');
|
| 2573 |
-
}
|
| 2574 |
-
/**
|
| 2575 |
-
* Enter description here...
|
| 2576 |
-
*
|
| 2577 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2578 |
-
*/
|
| 2579 |
-
public function submit($callback = null) {
|
| 2580 |
-
if ($callback)
|
| 2581 |
-
return $this->bind('submit', $callback);
|
| 2582 |
-
return $this->trigger('submit');
|
| 2583 |
-
}
|
| 2584 |
-
/**
|
| 2585 |
-
* Enter description here...
|
| 2586 |
-
*
|
| 2587 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2588 |
-
*/
|
| 2589 |
-
public function click($callback = null) {
|
| 2590 |
-
if ($callback)
|
| 2591 |
-
return $this->bind('click', $callback);
|
| 2592 |
-
return $this->trigger('click');
|
| 2593 |
-
}
|
| 2594 |
-
/**
|
| 2595 |
-
* Enter description here...
|
| 2596 |
-
*
|
| 2597 |
-
* @param String|phpQuery
|
| 2598 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2599 |
-
*/
|
| 2600 |
-
public function wrapAllOld($wrapper) {
|
| 2601 |
-
$wrapper = pq($wrapper)->_clone();
|
| 2602 |
-
if (! $wrapper->length() || ! $this->length() )
|
| 2603 |
-
return $this;
|
| 2604 |
-
$wrapper->insertBefore($this->elements[0]);
|
| 2605 |
-
$deepest = $wrapper->elements[0];
|
| 2606 |
-
while($deepest->firstChild && $deepest->firstChild instanceof DOMELEMENT)
|
| 2607 |
-
$deepest = $deepest->firstChild;
|
| 2608 |
-
pq($deepest)->append($this);
|
| 2609 |
-
return $this;
|
| 2610 |
-
}
|
| 2611 |
-
/**
|
| 2612 |
-
* Enter description here...
|
| 2613 |
-
*
|
| 2614 |
-
* TODO testme...
|
| 2615 |
-
* @param String|phpQuery
|
| 2616 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2617 |
-
*/
|
| 2618 |
-
public function wrapAll($wrapper) {
|
| 2619 |
-
if (! $this->length())
|
| 2620 |
-
return $this;
|
| 2621 |
-
return phpQuery::pq($wrapper, $this->getDocumentID())
|
| 2622 |
-
->clone()
|
| 2623 |
-
->insertBefore($this->get(0))
|
| 2624 |
-
->map(array($this, '___wrapAllCallback'))
|
| 2625 |
-
->append($this);
|
| 2626 |
-
}
|
| 2627 |
-
/**
|
| 2628 |
-
*
|
| 2629 |
-
* @param $node
|
| 2630 |
-
* @return unknown_type
|
| 2631 |
-
* @access private
|
| 2632 |
-
*/
|
| 2633 |
-
public function ___wrapAllCallback($node) {
|
| 2634 |
-
$deepest = $node;
|
| 2635 |
-
while($deepest->firstChild && $deepest->firstChild instanceof DOMELEMENT)
|
| 2636 |
-
$deepest = $deepest->firstChild;
|
| 2637 |
-
return $deepest;
|
| 2638 |
-
}
|
| 2639 |
-
/**
|
| 2640 |
-
* Enter description here...
|
| 2641 |
-
* NON JQUERY METHOD
|
| 2642 |
-
*
|
| 2643 |
-
* @param String|phpQuery
|
| 2644 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2645 |
-
*/
|
| 2646 |
-
public function wrapAllPHP($codeBefore, $codeAfter) {
|
| 2647 |
-
return $this
|
| 2648 |
-
->slice(0, 1)
|
| 2649 |
-
->beforePHP($codeBefore)
|
| 2650 |
-
->end()
|
| 2651 |
-
->slice(-1)
|
| 2652 |
-
->afterPHP($codeAfter)
|
| 2653 |
-
->end();
|
| 2654 |
-
}
|
| 2655 |
-
/**
|
| 2656 |
-
* Enter description here...
|
| 2657 |
-
*
|
| 2658 |
-
* @param String|phpQuery
|
| 2659 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2660 |
-
*/
|
| 2661 |
-
public function wrap($wrapper) {
|
| 2662 |
-
foreach($this->stack() as $node)
|
| 2663 |
-
phpQuery::pq($node, $this->getDocumentID())->wrapAll($wrapper);
|
| 2664 |
-
return $this;
|
| 2665 |
-
}
|
| 2666 |
-
/**
|
| 2667 |
-
* Enter description here...
|
| 2668 |
-
*
|
| 2669 |
-
* @param String|phpQuery
|
| 2670 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2671 |
-
*/
|
| 2672 |
-
public function wrapPHP($codeBefore, $codeAfter) {
|
| 2673 |
-
foreach($this->stack() as $node)
|
| 2674 |
-
phpQuery::pq($node, $this->getDocumentID())->wrapAllPHP($codeBefore, $codeAfter);
|
| 2675 |
-
return $this;
|
| 2676 |
-
}
|
| 2677 |
-
/**
|
| 2678 |
-
* Enter description here...
|
| 2679 |
-
*
|
| 2680 |
-
* @param String|phpQuery
|
| 2681 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2682 |
-
*/
|
| 2683 |
-
public function wrapInner($wrapper) {
|
| 2684 |
-
foreach($this->stack() as $node)
|
| 2685 |
-
phpQuery::pq($node, $this->getDocumentID())->contents()->wrapAll($wrapper);
|
| 2686 |
-
return $this;
|
| 2687 |
-
}
|
| 2688 |
-
/**
|
| 2689 |
-
* Enter description here...
|
| 2690 |
-
*
|
| 2691 |
-
* @param String|phpQuery
|
| 2692 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2693 |
-
*/
|
| 2694 |
-
public function wrapInnerPHP($codeBefore, $codeAfter) {
|
| 2695 |
-
foreach($this->stack(1) as $node)
|
| 2696 |
-
phpQuery::pq($node, $this->getDocumentID())->contents()
|
| 2697 |
-
->wrapAllPHP($codeBefore, $codeAfter);
|
| 2698 |
-
return $this;
|
| 2699 |
-
}
|
| 2700 |
-
/**
|
| 2701 |
-
* Enter description here...
|
| 2702 |
-
*
|
| 2703 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2704 |
-
* @testme Support for text nodes
|
| 2705 |
-
*/
|
| 2706 |
-
public function contents() {
|
| 2707 |
-
$stack = array();
|
| 2708 |
-
foreach($this->stack(1) as $el) {
|
| 2709 |
-
// FIXME (fixed) http://code.google.com/p/phpquery/issues/detail?id=56
|
| 2710 |
-
// if (! isset($el->childNodes))
|
| 2711 |
-
// continue;
|
| 2712 |
-
foreach($el->childNodes as $node) {
|
| 2713 |
-
$stack[] = $node;
|
| 2714 |
-
}
|
| 2715 |
-
}
|
| 2716 |
-
return $this->newInstance($stack);
|
| 2717 |
-
}
|
| 2718 |
-
/**
|
| 2719 |
-
* Enter description here...
|
| 2720 |
-
*
|
| 2721 |
-
* jQuery difference.
|
| 2722 |
-
*
|
| 2723 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2724 |
-
*/
|
| 2725 |
-
public function contentsUnwrap() {
|
| 2726 |
-
foreach($this->stack(1) as $node) {
|
| 2727 |
-
if (! $node->parentNode )
|
| 2728 |
-
continue;
|
| 2729 |
-
$childNodes = array();
|
| 2730 |
-
// any modification in DOM tree breaks childNodes iteration, so cache them first
|
| 2731 |
-
foreach($node->childNodes as $chNode )
|
| 2732 |
-
$childNodes[] = $chNode;
|
| 2733 |
-
foreach($childNodes as $chNode )
|
| 2734 |
-
// $node->parentNode->appendChild($chNode);
|
| 2735 |
-
$node->parentNode->insertBefore($chNode, $node);
|
| 2736 |
-
$node->parentNode->removeChild($node);
|
| 2737 |
-
}
|
| 2738 |
-
return $this;
|
| 2739 |
-
}
|
| 2740 |
-
/**
|
| 2741 |
-
* Enter description here...
|
| 2742 |
-
*
|
| 2743 |
-
* jQuery difference.
|
| 2744 |
-
*/
|
| 2745 |
-
public function switchWith($markup) {
|
| 2746 |
-
$markup = pq($markup, $this->getDocumentID());
|
| 2747 |
-
$content = null;
|
| 2748 |
-
foreach($this->stack(1) as $node) {
|
| 2749 |
-
pq($node)
|
| 2750 |
-
->contents()->toReference($content)->end()
|
| 2751 |
-
->replaceWith($markup->clone()->append($content));
|
| 2752 |
-
}
|
| 2753 |
-
return $this;
|
| 2754 |
-
}
|
| 2755 |
-
/**
|
| 2756 |
-
* Enter description here...
|
| 2757 |
-
*
|
| 2758 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2759 |
-
*/
|
| 2760 |
-
public function eq($num) {
|
| 2761 |
-
$oldStack = $this->elements;
|
| 2762 |
-
$this->elementsBackup = $this->elements;
|
| 2763 |
-
$this->elements = array();
|
| 2764 |
-
if ( isset($oldStack[$num]) )
|
| 2765 |
-
$this->elements[] = $oldStack[$num];
|
| 2766 |
-
return $this->newInstance();
|
| 2767 |
-
}
|
| 2768 |
-
/**
|
| 2769 |
-
* Enter description here...
|
| 2770 |
-
*
|
| 2771 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2772 |
-
*/
|
| 2773 |
-
public function size() {
|
| 2774 |
-
return count($this->elements);
|
| 2775 |
-
}
|
| 2776 |
-
/**
|
| 2777 |
-
* Enter description here...
|
| 2778 |
-
*
|
| 2779 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2780 |
-
* @deprecated Use length as attribute
|
| 2781 |
-
*/
|
| 2782 |
-
public function length() {
|
| 2783 |
-
return $this->size();
|
| 2784 |
-
}
|
| 2785 |
-
public function count() {
|
| 2786 |
-
return $this->size();
|
| 2787 |
-
}
|
| 2788 |
-
/**
|
| 2789 |
-
* Enter description here...
|
| 2790 |
-
*
|
| 2791 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2792 |
-
* @todo $level
|
| 2793 |
-
*/
|
| 2794 |
-
public function end($level = 1) {
|
| 2795 |
-
// $this->elements = array_pop( $this->history );
|
| 2796 |
-
// return $this;
|
| 2797 |
-
// $this->previous->DOM = $this->DOM;
|
| 2798 |
-
// $this->previous->XPath = $this->XPath;
|
| 2799 |
-
return $this->previous
|
| 2800 |
-
? $this->previous
|
| 2801 |
-
: $this;
|
| 2802 |
-
}
|
| 2803 |
-
/**
|
| 2804 |
-
* Enter description here...
|
| 2805 |
-
* Normal use ->clone() .
|
| 2806 |
-
*
|
| 2807 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2808 |
-
* @access private
|
| 2809 |
-
*/
|
| 2810 |
-
public function _clone() {
|
| 2811 |
-
$newStack = array();
|
| 2812 |
-
//pr(array('copy... ', $this->whois()));
|
| 2813 |
-
//$this->dumpHistory('copy');
|
| 2814 |
-
$this->elementsBackup = $this->elements;
|
| 2815 |
-
foreach($this->elements as $node) {
|
| 2816 |
-
$newStack[] = $node->cloneNode(true);
|
| 2817 |
-
}
|
| 2818 |
-
$this->elements = $newStack;
|
| 2819 |
-
return $this->newInstance();
|
| 2820 |
-
}
|
| 2821 |
-
/**
|
| 2822 |
-
* Enter description here...
|
| 2823 |
-
*
|
| 2824 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2825 |
-
*/
|
| 2826 |
-
public function replaceWithPHP($code) {
|
| 2827 |
-
return $this->replaceWith(phpQuery::php($code));
|
| 2828 |
-
}
|
| 2829 |
-
/**
|
| 2830 |
-
* Enter description here...
|
| 2831 |
-
*
|
| 2832 |
-
* @param String|phpQuery $content
|
| 2833 |
-
* @link http://docs.jquery.com/Manipulation/replaceWith#content
|
| 2834 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2835 |
-
*/
|
| 2836 |
-
public function replaceWith($content) {
|
| 2837 |
-
return $this->after($content)->remove();
|
| 2838 |
-
}
|
| 2839 |
-
/**
|
| 2840 |
-
* Enter description here...
|
| 2841 |
-
*
|
| 2842 |
-
* @param String $selector
|
| 2843 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2844 |
-
* @todo this works ?
|
| 2845 |
-
*/
|
| 2846 |
-
public function replaceAll($selector) {
|
| 2847 |
-
foreach(phpQuery::pq($selector, $this->getDocumentID()) as $node)
|
| 2848 |
-
phpQuery::pq($node, $this->getDocumentID())
|
| 2849 |
-
->after($this->_clone())
|
| 2850 |
-
->remove();
|
| 2851 |
-
return $this;
|
| 2852 |
-
}
|
| 2853 |
-
/**
|
| 2854 |
-
* Enter description here...
|
| 2855 |
-
*
|
| 2856 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2857 |
-
*/
|
| 2858 |
-
public function remove($selector = null) {
|
| 2859 |
-
$loop = $selector
|
| 2860 |
-
? $this->filter($selector)->elements
|
| 2861 |
-
: $this->elements;
|
| 2862 |
-
foreach($loop as $node) {
|
| 2863 |
-
if (! $node->parentNode )
|
| 2864 |
-
continue;
|
| 2865 |
-
if (isset($node->tagName))
|
| 2866 |
-
$this->debug("Removing '{$node->tagName}'");
|
| 2867 |
-
$node->parentNode->removeChild($node);
|
| 2868 |
-
// Mutation event
|
| 2869 |
-
$event = new DOMEvent(array(
|
| 2870 |
-
'target' => $node,
|
| 2871 |
-
'type' => 'DOMNodeRemoved'
|
| 2872 |
-
));
|
| 2873 |
-
phpQueryEvents::trigger($this->getDocumentID(),
|
| 2874 |
-
$event->type, array($event), $node
|
| 2875 |
-
);
|
| 2876 |
-
}
|
| 2877 |
-
return $this;
|
| 2878 |
-
}
|
| 2879 |
-
protected function markupEvents($newMarkup, $oldMarkup, $node) {
|
| 2880 |
-
if ($node->tagName == 'textarea' && $newMarkup != $oldMarkup) {
|
| 2881 |
-
$event = new DOMEvent(array(
|
| 2882 |
-
'target' => $node,
|
| 2883 |
-
'type' => 'change'
|
| 2884 |
-
));
|
| 2885 |
-
phpQueryEvents::trigger($this->getDocumentID(),
|
| 2886 |
-
$event->type, array($event), $node
|
| 2887 |
-
);
|
| 2888 |
-
}
|
| 2889 |
-
}
|
| 2890 |
-
/**
|
| 2891 |
-
* jQuey difference
|
| 2892 |
-
*
|
| 2893 |
-
* @param $markup
|
| 2894 |
-
* @return unknown_type
|
| 2895 |
-
* @TODO trigger change event for textarea
|
| 2896 |
-
*/
|
| 2897 |
-
public function markup($markup = null, $callback1 = null, $callback2 = null, $callback3 = null) {
|
| 2898 |
-
$args = func_get_args();
|
| 2899 |
-
if ($this->documentWrapper->isXML)
|
| 2900 |
-
return call_user_func_array(array($this, 'xml'), $args);
|
| 2901 |
-
else
|
| 2902 |
-
return call_user_func_array(array($this, 'html'), $args);
|
| 2903 |
-
}
|
| 2904 |
-
/**
|
| 2905 |
-
* jQuey difference
|
| 2906 |
-
*
|
| 2907 |
-
* @param $markup
|
| 2908 |
-
* @return unknown_type
|
| 2909 |
-
*/
|
| 2910 |
-
public function markupOuter($callback1 = null, $callback2 = null, $callback3 = null) {
|
| 2911 |
-
$args = func_get_args();
|
| 2912 |
-
if ($this->documentWrapper->isXML)
|
| 2913 |
-
return call_user_func_array(array($this, 'xmlOuter'), $args);
|
| 2914 |
-
else
|
| 2915 |
-
return call_user_func_array(array($this, 'htmlOuter'), $args);
|
| 2916 |
-
}
|
| 2917 |
-
/**
|
| 2918 |
-
* Enter description here...
|
| 2919 |
-
*
|
| 2920 |
-
* @param unknown_type $html
|
| 2921 |
-
* @return string|phpQuery|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2922 |
-
* @TODO force html result
|
| 2923 |
-
*/
|
| 2924 |
-
public function html($html = null, $callback1 = null, $callback2 = null, $callback3 = null) {
|
| 2925 |
-
if (isset($html)) {
|
| 2926 |
-
// INSERT
|
| 2927 |
-
$nodes = $this->documentWrapper->import($html);
|
| 2928 |
-
$this->empty();
|
| 2929 |
-
foreach($this->stack(1) as $alreadyAdded => $node) {
|
| 2930 |
-
// for now, limit events for textarea
|
| 2931 |
-
if (($this->isXHTML() || $this->isHTML()) && $node->tagName == 'textarea')
|
| 2932 |
-
$oldHtml = pq($node, $this->getDocumentID())->markup();
|
| 2933 |
-
foreach($nodes as $newNode) {
|
| 2934 |
-
$node->appendChild($alreadyAdded
|
| 2935 |
-
? $newNode->cloneNode(true)
|
| 2936 |
-
: $newNode
|
| 2937 |
-
);
|
| 2938 |
-
}
|
| 2939 |
-
// for now, limit events for textarea
|
| 2940 |
-
if (($this->isXHTML() || $this->isHTML()) && $node->tagName == 'textarea')
|
| 2941 |
-
$this->markupEvents($html, $oldHtml, $node);
|
| 2942 |
-
}
|
| 2943 |
-
return $this;
|
| 2944 |
-
} else {
|
| 2945 |
-
// FETCH
|
| 2946 |
-
$return = $this->documentWrapper->markup($this->elements, true);
|
| 2947 |
-
$args = func_get_args();
|
| 2948 |
-
foreach(array_slice($args, 1) as $callback) {
|
| 2949 |
-
$return = phpQuery::callbackRun($callback, array($return));
|
| 2950 |
-
}
|
| 2951 |
-
return $return;
|
| 2952 |
-
}
|
| 2953 |
-
}
|
| 2954 |
-
/**
|
| 2955 |
-
* @TODO force xml result
|
| 2956 |
-
*/
|
| 2957 |
-
public function xml($xml = null, $callback1 = null, $callback2 = null, $callback3 = null) {
|
| 2958 |
-
$args = func_get_args();
|
| 2959 |
-
return call_user_func_array(array($this, 'html'), $args);
|
| 2960 |
-
}
|
| 2961 |
-
/**
|
| 2962 |
-
* Enter description here...
|
| 2963 |
-
* @TODO force html result
|
| 2964 |
-
*
|
| 2965 |
-
* @return String
|
| 2966 |
-
*/
|
| 2967 |
-
public function htmlOuter($callback1 = null, $callback2 = null, $callback3 = null) {
|
| 2968 |
-
$markup = $this->documentWrapper->markup($this->elements);
|
| 2969 |
-
// pass thou callbacks
|
| 2970 |
-
$args = func_get_args();
|
| 2971 |
-
foreach($args as $callback) {
|
| 2972 |
-
$markup = phpQuery::callbackRun($callback, array($markup));
|
| 2973 |
-
}
|
| 2974 |
-
return $markup;
|
| 2975 |
-
}
|
| 2976 |
-
/**
|
| 2977 |
-
* @TODO force xml result
|
| 2978 |
-
*/
|
| 2979 |
-
public function xmlOuter($callback1 = null, $callback2 = null, $callback3 = null) {
|
| 2980 |
-
$args = func_get_args();
|
| 2981 |
-
return call_user_func_array(array($this, 'htmlOuter'), $args);
|
| 2982 |
-
}
|
| 2983 |
-
public function __toString() {
|
| 2984 |
-
return $this->markupOuter();
|
| 2985 |
-
}
|
| 2986 |
-
/**
|
| 2987 |
-
* Just like html(), but returns markup with VALID (dangerous) PHP tags.
|
| 2988 |
-
*
|
| 2989 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2990 |
-
* @todo support returning markup with PHP tags when called without param
|
| 2991 |
-
*/
|
| 2992 |
-
public function php($code = null) {
|
| 2993 |
-
return $this->markupPHP($code);
|
| 2994 |
-
}
|
| 2995 |
-
/**
|
| 2996 |
-
* Enter description here...
|
| 2997 |
-
*
|
| 2998 |
-
* @param $code
|
| 2999 |
-
* @return unknown_type
|
| 3000 |
-
*/
|
| 3001 |
-
public function markupPHP($code = null) {
|
| 3002 |
-
return isset($code)
|
| 3003 |
-
? $this->markup(phpQuery::php($code))
|
| 3004 |
-
: phpQuery::markupToPHP($this->markup());
|
| 3005 |
-
}
|
| 3006 |
-
/**
|
| 3007 |
-
* Enter description here...
|
| 3008 |
-
*
|
| 3009 |
-
* @param $code
|
| 3010 |
-
* @return unknown_type
|
| 3011 |
-
*/
|
| 3012 |
-
public function markupOuterPHP() {
|
| 3013 |
-
return phpQuery::markupToPHP($this->markupOuter());
|
| 3014 |
-
}
|
| 3015 |
-
/**
|
| 3016 |
-
* Enter description here...
|
| 3017 |
-
*
|
| 3018 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3019 |
-
*/
|
| 3020 |
-
public function children($selector = null) {
|
| 3021 |
-
$stack = array();
|
| 3022 |
-
foreach($this->stack(1) as $node) {
|
| 3023 |
-
// foreach($node->getElementsByTagName('*') as $newNode) {
|
| 3024 |
-
foreach($node->childNodes as $newNode) {
|
| 3025 |
-
if ($newNode->nodeType != 1)
|
| 3026 |
-
continue;
|
| 3027 |
-
if ($selector && ! $this->is($selector, $newNode))
|
| 3028 |
-
continue;
|
| 3029 |
-
if ($this->elementsContainsNode($newNode, $stack))
|
| 3030 |
-
continue;
|
| 3031 |
-
$stack[] = $newNode;
|
| 3032 |
-
}
|
| 3033 |
-
}
|
| 3034 |
-
$this->elementsBackup = $this->elements;
|
| 3035 |
-
$this->elements = $stack;
|
| 3036 |
-
return $this->newInstance();
|
| 3037 |
-
}
|
| 3038 |
-
/**
|
| 3039 |
-
* Enter description here...
|
| 3040 |
-
*
|
| 3041 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3042 |
-
*/
|
| 3043 |
-
public function ancestors($selector = null) {
|
| 3044 |
-
return $this->children( $selector );
|
| 3045 |
-
}
|
| 3046 |
-
/**
|
| 3047 |
-
* Enter description here...
|
| 3048 |
-
*
|
| 3049 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3050 |
-
*/
|
| 3051 |
-
public function append( $content) {
|
| 3052 |
-
return $this->insert($content, __FUNCTION__);
|
| 3053 |
-
}
|
| 3054 |
-
/**
|
| 3055 |
-
* Enter description here...
|
| 3056 |
-
*
|
| 3057 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3058 |
-
*/
|
| 3059 |
-
public function appendPHP( $content) {
|
| 3060 |
-
return $this->insert("<php><!-- {$content} --></php>", 'append');
|
| 3061 |
-
}
|
| 3062 |
-
/**
|
| 3063 |
-
* Enter description here...
|
| 3064 |
-
*
|
| 3065 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3066 |
-
*/
|
| 3067 |
-
public function appendTo( $seletor) {
|
| 3068 |
-
return $this->insert($seletor, __FUNCTION__);
|
| 3069 |
-
}
|
| 3070 |
-
/**
|
| 3071 |
-
* Enter description here...
|
| 3072 |
-
*
|
| 3073 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3074 |
-
*/
|
| 3075 |
-
public function prepend( $content) {
|
| 3076 |
-
return $this->insert($content, __FUNCTION__);
|
| 3077 |
-
}
|
| 3078 |
-
/**
|
| 3079 |
-
* Enter description here...
|
| 3080 |
-
*
|
| 3081 |
-
* @todo accept many arguments, which are joined, arrays maybe also
|
| 3082 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3083 |
-
*/
|
| 3084 |
-
public function prependPHP( $content) {
|
| 3085 |
-
return $this->insert("<php><!-- {$content} --></php>", 'prepend');
|
| 3086 |
-
}
|
| 3087 |
-
/**
|
| 3088 |
-
* Enter description here...
|
| 3089 |
-
*
|
| 3090 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3091 |
-
*/
|
| 3092 |
-
public function prependTo( $seletor) {
|
| 3093 |
-
return $this->insert($seletor, __FUNCTION__);
|
| 3094 |
-
}
|
| 3095 |
-
/**
|
| 3096 |
-
* Enter description here...
|
| 3097 |
-
*
|
| 3098 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3099 |
-
*/
|
| 3100 |
-
public function before($content) {
|
| 3101 |
-
return $this->insert($content, __FUNCTION__);
|
| 3102 |
-
}
|
| 3103 |
-
/**
|
| 3104 |
-
* Enter description here...
|
| 3105 |
-
*
|
| 3106 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3107 |
-
*/
|
| 3108 |
-
public function beforePHP( $content) {
|
| 3109 |
-
return $this->insert("<php><!-- {$content} --></php>", 'before');
|
| 3110 |
-
}
|
| 3111 |
-
/**
|
| 3112 |
-
* Enter description here...
|
| 3113 |
-
*
|
| 3114 |
-
* @param String|phpQuery
|
| 3115 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3116 |
-
*/
|
| 3117 |
-
public function insertBefore( $seletor) {
|
| 3118 |
-
return $this->insert($seletor, __FUNCTION__);
|
| 3119 |
-
}
|
| 3120 |
-
/**
|
| 3121 |
-
* Enter description here...
|
| 3122 |
-
*
|
| 3123 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3124 |
-
*/
|
| 3125 |
-
public function after( $content) {
|
| 3126 |
-
return $this->insert($content, __FUNCTION__);
|
| 3127 |
-
}
|
| 3128 |
-
/**
|
| 3129 |
-
* Enter description here...
|
| 3130 |
-
*
|
| 3131 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3132 |
-
*/
|
| 3133 |
-
public function afterPHP( $content) {
|
| 3134 |
-
return $this->insert("<php><!-- {$content} --></php>", 'after');
|
| 3135 |
-
}
|
| 3136 |
-
/**
|
| 3137 |
-
* Enter description here...
|
| 3138 |
-
*
|
| 3139 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3140 |
-
*/
|
| 3141 |
-
public function insertAfter( $seletor) {
|
| 3142 |
-
return $this->insert($seletor, __FUNCTION__);
|
| 3143 |
-
}
|
| 3144 |
-
/**
|
| 3145 |
-
* Internal insert method. Don't use it.
|
| 3146 |
-
*
|
| 3147 |
-
* @param unknown_type $target
|
| 3148 |
-
* @param unknown_type $type
|
| 3149 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3150 |
-
* @access private
|
| 3151 |
-
*/
|
| 3152 |
-
public function insert($target, $type) {
|
| 3153 |
-
$this->debug("Inserting data with '{$type}'");
|
| 3154 |
-
$to = false;
|
| 3155 |
-
switch( $type) {
|
| 3156 |
-
case 'appendTo':
|
| 3157 |
-
case 'prependTo':
|
| 3158 |
-
case 'insertBefore':
|
| 3159 |
-
case 'insertAfter':
|
| 3160 |
-
$to = true;
|
| 3161 |
-
}
|
| 3162 |
-
switch(gettype($target)) {
|
| 3163 |
-
case 'string':
|
| 3164 |
-
$insertFrom = $insertTo = array();
|
| 3165 |
-
if ($to) {
|
| 3166 |
-
// INSERT TO
|
| 3167 |
-
$insertFrom = $this->elements;
|
| 3168 |
-
if (phpQuery::isMarkup($target)) {
|
| 3169 |
-
// $target is new markup, import it
|
| 3170 |
-
$insertTo = $this->documentWrapper->import($target);
|
| 3171 |
-
// insert into selected element
|
| 3172 |
-
} else {
|
| 3173 |
-
// $tagret is a selector
|
| 3174 |
-
$thisStack = $this->elements;
|
| 3175 |
-
$this->toRoot();
|
| 3176 |
-
$insertTo = $this->find($target)->elements;
|
| 3177 |
-
$this->elements = $thisStack;
|
| 3178 |
-
}
|
| 3179 |
-
} else {
|
| 3180 |
-
// INSERT FROM
|
| 3181 |
-
$insertTo = $this->elements;
|
| 3182 |
-
$insertFrom = $this->documentWrapper->import($target);
|
| 3183 |
-
}
|
| 3184 |
-
break;
|
| 3185 |
-
case 'object':
|
| 3186 |
-
$insertFrom = $insertTo = array();
|
| 3187 |
-
// phpQuery
|
| 3188 |
-
if ($target instanceof self) {
|
| 3189 |
-
if ($to) {
|
| 3190 |
-
$insertTo = $target->elements;
|
| 3191 |
-
if ($this->documentFragment && $this->stackIsRoot())
|
| 3192 |
-
// get all body children
|
| 3193 |
-
// $loop = $this->find('body > *')->elements;
|
| 3194 |
-
// TODO test it, test it hard...
|
| 3195 |
-
// $loop = $this->newInstance($this->root)->find('> *')->elements;
|
| 3196 |
-
$loop = $this->root->childNodes;
|
| 3197 |
-
else
|
| 3198 |
-
$loop = $this->elements;
|
| 3199 |
-
// import nodes if needed
|
| 3200 |
-
$insertFrom = $this->getDocumentID() == $target->getDocumentID()
|
| 3201 |
-
? $loop
|
| 3202 |
-
: $target->documentWrapper->import($loop);
|
| 3203 |
-
} else {
|
| 3204 |
-
$insertTo = $this->elements;
|
| 3205 |
-
if ( $target->documentFragment && $target->stackIsRoot() )
|
| 3206 |
-
// get all body children
|
| 3207 |
-
// $loop = $target->find('body > *')->elements;
|
| 3208 |
-
$loop = $target->root->childNodes;
|
| 3209 |
-
else
|
| 3210 |
-
$loop = $target->elements;
|
| 3211 |
-
// import nodes if needed
|
| 3212 |
-
$insertFrom = $this->getDocumentID() == $target->getDocumentID()
|
| 3213 |
-
? $loop
|
| 3214 |
-
: $this->documentWrapper->import($loop);
|
| 3215 |
-
}
|
| 3216 |
-
// DOMNODE
|
| 3217 |
-
} elseif ($target instanceof DOMNODE) {
|
| 3218 |
-
// import node if needed
|
| 3219 |
-
// if ( $target->ownerDocument != $this->DOM )
|
| 3220 |
-
// $target = $this->DOM->importNode($target, true);
|
| 3221 |
-
if ( $to) {
|
| 3222 |
-
$insertTo = array($target);
|
| 3223 |
-
if ($this->documentFragment && $this->stackIsRoot())
|
| 3224 |
-
// get all body children
|
| 3225 |
-
$loop = $this->root->childNodes;
|
| 3226 |
-
// $loop = $this->find('body > *')->elements;
|
| 3227 |
-
else
|
| 3228 |
-
$loop = $this->elements;
|
| 3229 |
-
foreach($loop as $fromNode)
|
| 3230 |
-
// import nodes if needed
|
| 3231 |
-
$insertFrom[] = ! $fromNode->ownerDocument->isSameNode($target->ownerDocument)
|
| 3232 |
-
? $target->ownerDocument->importNode($fromNode, true)
|
| 3233 |
-
: $fromNode;
|
| 3234 |
-
} else {
|
| 3235 |
-
// import node if needed
|
| 3236 |
-
if (! $target->ownerDocument->isSameNode($this->document))
|
| 3237 |
-
$target = $this->document->importNode($target, true);
|
| 3238 |
-
$insertTo = $this->elements;
|
| 3239 |
-
$insertFrom[] = $target;
|
| 3240 |
-
}
|
| 3241 |
-
}
|
| 3242 |
-
break;
|
| 3243 |
-
}
|
| 3244 |
-
phpQuery::debug("From ".count($insertFrom)."; To ".count($insertTo)." nodes");
|
| 3245 |
-
foreach($insertTo as $insertNumber => $toNode) {
|
| 3246 |
-
// we need static relative elements in some cases
|
| 3247 |
-
switch( $type) {
|
| 3248 |
-
case 'prependTo':
|
| 3249 |
-
case 'prepend':
|
| 3250 |
-
$firstChild = $toNode->firstChild;
|
| 3251 |
-
break;
|
| 3252 |
-
case 'insertAfter':
|
| 3253 |
-
case 'after':
|
| 3254 |
-
$nextSibling = $toNode->nextSibling;
|
| 3255 |
-
break;
|
| 3256 |
-
}
|
| 3257 |
-
foreach($insertFrom as $fromNode) {
|
| 3258 |
-
// clone if inserted already before
|
| 3259 |
-
$insert = $insertNumber
|
| 3260 |
-
? $fromNode->cloneNode(true)
|
| 3261 |
-
: $fromNode;
|
| 3262 |
-
switch($type) {
|
| 3263 |
-
case 'appendTo':
|
| 3264 |
-
case 'append':
|
| 3265 |
-
// $toNode->insertBefore(
|
| 3266 |
-
// $fromNode,
|
| 3267 |
-
// $toNode->lastChild->nextSibling
|
| 3268 |
-
// );
|
| 3269 |
-
$toNode->appendChild($insert);
|
| 3270 |
-
$eventTarget = $insert;
|
| 3271 |
-
break;
|
| 3272 |
-
case 'prependTo':
|
| 3273 |
-
case 'prepend':
|
| 3274 |
-
$toNode->insertBefore(
|
| 3275 |
-
$insert,
|
| 3276 |
-
$firstChild
|
| 3277 |
-
);
|
| 3278 |
-
break;
|
| 3279 |
-
case 'insertBefore':
|
| 3280 |
-
case 'before':
|
| 3281 |
-
if (! $toNode->parentNode)
|
| 3282 |
-
throw new Exception("No parentNode, can't do {$type}()");
|
| 3283 |
-
else
|
| 3284 |
-
$toNode->parentNode->insertBefore(
|
| 3285 |
-
$insert,
|
| 3286 |
-
$toNode
|
| 3287 |
-
);
|
| 3288 |
-
break;
|
| 3289 |
-
case 'insertAfter':
|
| 3290 |
-
case 'after':
|
| 3291 |
-
if (! $toNode->parentNode)
|
| 3292 |
-
throw new Exception("No parentNode, can't do {$type}()");
|
| 3293 |
-
else
|
| 3294 |
-
$toNode->parentNode->insertBefore(
|
| 3295 |
-
$insert,
|
| 3296 |
-
$nextSibling
|
| 3297 |
-
);
|
| 3298 |
-
break;
|
| 3299 |
-
}
|
| 3300 |
-
// Mutation event
|
| 3301 |
-
$event = new DOMEvent(array(
|
| 3302 |
-
'target' => $insert,
|
| 3303 |
-
'type' => 'DOMNodeInserted'
|
| 3304 |
-
));
|
| 3305 |
-
phpQueryEvents::trigger($this->getDocumentID(),
|
| 3306 |
-
$event->type, array($event), $insert
|
| 3307 |
-
);
|
| 3308 |
-
}
|
| 3309 |
-
}
|
| 3310 |
-
return $this;
|
| 3311 |
-
}
|
| 3312 |
-
/**
|
| 3313 |
-
* Enter description here...
|
| 3314 |
-
*
|
| 3315 |
-
* @return Int
|
| 3316 |
-
*/
|
| 3317 |
-
public function index($subject) {
|
| 3318 |
-
$index = -1;
|
| 3319 |
-
$subject = $subject instanceof phpQueryObject
|
| 3320 |
-
? $subject->elements[0]
|
| 3321 |
-
: $subject;
|
| 3322 |
-
foreach($this->newInstance() as $k => $node) {
|
| 3323 |
-
if ($node->isSameNode($subject))
|
| 3324 |
-
$index = $k;
|
| 3325 |
-
}
|
| 3326 |
-
return $index;
|
| 3327 |
-
}
|
| 3328 |
-
/**
|
| 3329 |
-
* Enter description here...
|
| 3330 |
-
*
|
| 3331 |
-
* @param unknown_type $start
|
| 3332 |
-
* @param unknown_type $end
|
| 3333 |
-
*
|
| 3334 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3335 |
-
* @testme
|
| 3336 |
-
*/
|
| 3337 |
-
public function slice($start, $end = null) {
|
| 3338 |
-
// $last = count($this->elements)-1;
|
| 3339 |
-
// $end = $end
|
| 3340 |
-
// ? min($end, $last)
|
| 3341 |
-
// : $last;
|
| 3342 |
-
// if ($start < 0)
|
| 3343 |
-
// $start = $last+$start;
|
| 3344 |
-
// if ($start > $last)
|
| 3345 |
-
// return array();
|
| 3346 |
-
if ($end > 0)
|
| 3347 |
-
$end = $end-$start;
|
| 3348 |
-
return $this->newInstance(
|
| 3349 |
-
array_slice($this->elements, $start, $end)
|
| 3350 |
-
);
|
| 3351 |
-
}
|
| 3352 |
-
/**
|
| 3353 |
-
* Enter description here...
|
| 3354 |
-
*
|
| 3355 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3356 |
-
*/
|
| 3357 |
-
public function reverse() {
|
| 3358 |
-
$this->elementsBackup = $this->elements;
|
| 3359 |
-
$this->elements = array_reverse($this->elements);
|
| 3360 |
-
return $this->newInstance();
|
| 3361 |
-
}
|
| 3362 |
-
/**
|
| 3363 |
-
* Return joined text content.
|
| 3364 |
-
* @return String
|
| 3365 |
-
*/
|
| 3366 |
-
public function text($text = null, $callback1 = null, $callback2 = null, $callback3 = null) {
|
| 3367 |
-
if (isset($text))
|
| 3368 |
-
return $this->html(htmlspecialchars($text));
|
| 3369 |
-
$args = func_get_args();
|
| 3370 |
-
$args = array_slice($args, 1);
|
| 3371 |
-
$return = '';
|
| 3372 |
-
foreach($this->elements as $node) {
|
| 3373 |
-
$text = $node->textContent;
|
| 3374 |
-
if (count($this->elements) > 1 && $text)
|
| 3375 |
-
$text .= "\n";
|
| 3376 |
-
foreach($args as $callback) {
|
| 3377 |
-
$text = phpQuery::callbackRun($callback, array($text));
|
| 3378 |
-
}
|
| 3379 |
-
$return .= $text;
|
| 3380 |
-
}
|
| 3381 |
-
return $return;
|
| 3382 |
-
}
|
| 3383 |
-
/**
|
| 3384 |
-
* Enter description here...
|
| 3385 |
-
*
|
| 3386 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3387 |
-
*/
|
| 3388 |
-
public function plugin($class, $file = null) {
|
| 3389 |
-
phpQuery::plugin($class, $file);
|
| 3390 |
-
return $this;
|
| 3391 |
-
}
|
| 3392 |
-
/**
|
| 3393 |
-
* Deprecated, use $pq->plugin() instead.
|
| 3394 |
-
*
|
| 3395 |
-
* @deprecated
|
| 3396 |
-
* @param $class
|
| 3397 |
-
* @param $file
|
| 3398 |
-
* @return unknown_type
|
| 3399 |
-
*/
|
| 3400 |
-
public static function extend($class, $file = null) {
|
| 3401 |
-
return $this->plugin($class, $file);
|
| 3402 |
-
}
|
| 3403 |
-
/**
|
| 3404 |
-
*
|
| 3405 |
-
* @access private
|
| 3406 |
-
* @param $method
|
| 3407 |
-
* @param $args
|
| 3408 |
-
* @return unknown_type
|
| 3409 |
-
*/
|
| 3410 |
-
public function __call($method, $args) {
|
| 3411 |
-
$aliasMethods = array('clone', 'empty');
|
| 3412 |
-
if (isset(phpQuery::$extendMethods[$method])) {
|
| 3413 |
-
array_unshift($args, $this);
|
| 3414 |
-
return phpQuery::callbackRun(
|
| 3415 |
-
phpQuery::$extendMethods[$method], $args
|
| 3416 |
-
);
|
| 3417 |
-
} else if (isset(phpQuery::$pluginsMethods[$method])) {
|
| 3418 |
-
array_unshift($args, $this);
|
| 3419 |
-
$class = phpQuery::$pluginsMethods[$method];
|
| 3420 |
-
$realClass = "phpQueryObjectPlugin_$class";
|
| 3421 |
-
$return = call_user_func_array(
|
| 3422 |
-
array($realClass, $method),
|
| 3423 |
-
$args
|
| 3424 |
-
);
|
| 3425 |
-
// XXX deprecate ?
|
| 3426 |
-
return is_null($return)
|
| 3427 |
-
? $this
|
| 3428 |
-
: $return;
|
| 3429 |
-
} else if (in_array($method, $aliasMethods)) {
|
| 3430 |
-
return call_user_func_array(array($this, '_'.$method), $args);
|
| 3431 |
-
} else
|
| 3432 |
-
throw new Exception("Method '{$method}' doesnt exist");
|
| 3433 |
-
}
|
| 3434 |
-
/**
|
| 3435 |
-
* Safe rename of next().
|
| 3436 |
-
*
|
| 3437 |
-
* Use it ONLY when need to call next() on an iterated object (in same time).
|
| 3438 |
-
* Normaly there is no need to do such thing ;)
|
| 3439 |
-
*
|
| 3440 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3441 |
-
* @access private
|
| 3442 |
-
*/
|
| 3443 |
-
public function _next($selector = null) {
|
| 3444 |
-
return $this->newInstance(
|
| 3445 |
-
$this->getElementSiblings('nextSibling', $selector, true)
|
| 3446 |
-
);
|
| 3447 |
-
}
|
| 3448 |
-
/**
|
| 3449 |
-
* Use prev() and next().
|
| 3450 |
-
*
|
| 3451 |
-
* @deprecated
|
| 3452 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3453 |
-
* @access private
|
| 3454 |
-
*/
|
| 3455 |
-
public function _prev($selector = null) {
|
| 3456 |
-
return $this->prev($selector);
|
| 3457 |
-
}
|
| 3458 |
-
/**
|
| 3459 |
-
* Enter description here...
|
| 3460 |
-
*
|
| 3461 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3462 |
-
*/
|
| 3463 |
-
public function prev($selector = null) {
|
| 3464 |
-
return $this->newInstance(
|
| 3465 |
-
$this->getElementSiblings('previousSibling', $selector, true)
|
| 3466 |
-
);
|
| 3467 |
-
}
|
| 3468 |
-
/**
|
| 3469 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3470 |
-
* @todo
|
| 3471 |
-
*/
|
| 3472 |
-
public function prevAll($selector = null) {
|
| 3473 |
-
return $this->newInstance(
|
| 3474 |
-
$this->getElementSiblings('previousSibling', $selector)
|
| 3475 |
-
);
|
| 3476 |
-
}
|
| 3477 |
-
/**
|
| 3478 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3479 |
-
* @todo FIXME: returns source elements insted of next siblings
|
| 3480 |
-
*/
|
| 3481 |
-
public function nextAll($selector = null) {
|
| 3482 |
-
return $this->newInstance(
|
| 3483 |
-
$this->getElementSiblings('nextSibling', $selector)
|
| 3484 |
-
);
|
| 3485 |
-
}
|
| 3486 |
-
/**
|
| 3487 |
-
* @access private
|
| 3488 |
-
*/
|
| 3489 |
-
protected function getElementSiblings($direction, $selector = null, $limitToOne = false) {
|
| 3490 |
-
$stack = array();
|
| 3491 |
-
$count = 0;
|
| 3492 |
-
foreach($this->stack() as $node) {
|
| 3493 |
-
$test = $node;
|
| 3494 |
-
while( isset($test->{$direction}) && $test->{$direction}) {
|
| 3495 |
-
$test = $test->{$direction};
|
| 3496 |
-
if (! $test instanceof DOMELEMENT)
|
| 3497 |
-
continue;
|
| 3498 |
-
$stack[] = $test;
|
| 3499 |
-
if ($limitToOne)
|
| 3500 |
-
break;
|
| 3501 |
-
}
|
| 3502 |
-
}
|
| 3503 |
-
if ($selector) {
|
| 3504 |
-
$stackOld = $this->elements;
|
| 3505 |
-
$this->elements = $stack;
|
| 3506 |
-
$stack = $this->filter($selector, true)->stack();
|
| 3507 |
-
$this->elements = $stackOld;
|
| 3508 |
-
}
|
| 3509 |
-
return $stack;
|
| 3510 |
-
}
|
| 3511 |
-
/**
|
| 3512 |
-
* Enter description here...
|
| 3513 |
-
*
|
| 3514 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3515 |
-
*/
|
| 3516 |
-
public function siblings($selector = null) {
|
| 3517 |
-
$stack = array();
|
| 3518 |
-
$siblings = array_merge(
|
| 3519 |
-
$this->getElementSiblings('previousSibling', $selector),
|
| 3520 |
-
$this->getElementSiblings('nextSibling', $selector)
|
| 3521 |
-
);
|
| 3522 |
-
foreach($siblings as $node) {
|
| 3523 |
-
if (! $this->elementsContainsNode($node, $stack))
|
| 3524 |
-
$stack[] = $node;
|
| 3525 |
-
}
|
| 3526 |
-
return $this->newInstance($stack);
|
| 3527 |
-
}
|
| 3528 |
-
/**
|
| 3529 |
-
* Enter description here...
|
| 3530 |
-
*
|
| 3531 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3532 |
-
*/
|
| 3533 |
-
public function not($selector = null) {
|
| 3534 |
-
if (is_string($selector))
|
| 3535 |
-
phpQuery::debug(array('not', $selector));
|
| 3536 |
-
else
|
| 3537 |
-
phpQuery::debug('not');
|
| 3538 |
-
$stack = array();
|
| 3539 |
-
if ($selector instanceof self || $selector instanceof DOMNODE) {
|
| 3540 |
-
foreach($this->stack() as $node) {
|
| 3541 |
-
if ($selector instanceof self) {
|
| 3542 |
-
$matchFound = false;
|
| 3543 |
-
foreach($selector->stack() as $notNode) {
|
| 3544 |
-
if ($notNode->isSameNode($node))
|
| 3545 |
-
$matchFound = true;
|
| 3546 |
-
}
|
| 3547 |
-
if (! $matchFound)
|
| 3548 |
-
$stack[] = $node;
|
| 3549 |
-
} else if ($selector instanceof DOMNODE) {
|
| 3550 |
-
if (! $selector->isSameNode($node))
|
| 3551 |
-
$stack[] = $node;
|
| 3552 |
-
} else {
|
| 3553 |
-
if (! $this->is($selector))
|
| 3554 |
-
$stack[] = $node;
|
| 3555 |
-
}
|
| 3556 |
-
}
|
| 3557 |
-
} else {
|
| 3558 |
-
$orgStack = $this->stack();
|
| 3559 |
-
$matched = $this->filter($selector, true)->stack();
|
| 3560 |
-
// $matched = array();
|
| 3561 |
-
// // simulate OR in filter() instead of AND 5y
|
| 3562 |
-
// foreach($this->parseSelector($selector) as $s) {
|
| 3563 |
-
// $matched = array_merge($matched,
|
| 3564 |
-
// $this->filter(array($s))->stack()
|
| 3565 |
-
// );
|
| 3566 |
-
// }
|
| 3567 |
-
foreach($orgStack as $node)
|
| 3568 |
-
if (! $this->elementsContainsNode($node, $matched))
|
| 3569 |
-
$stack[] = $node;
|
| 3570 |
-
}
|
| 3571 |
-
return $this->newInstance($stack);
|
| 3572 |
-
}
|
| 3573 |
-
/**
|
| 3574 |
-
* Enter description here...
|
| 3575 |
-
*
|
| 3576 |
-
* @param string|phpQueryObject
|
| 3577 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3578 |
-
*/
|
| 3579 |
-
public function add($selector = null) {
|
| 3580 |
-
if (! $selector)
|
| 3581 |
-
return $this;
|
| 3582 |
-
$stack = array();
|
| 3583 |
-
$this->elementsBackup = $this->elements;
|
| 3584 |
-
$found = phpQuery::pq($selector, $this->getDocumentID());
|
| 3585 |
-
$this->merge($found->elements);
|
| 3586 |
-
return $this->newInstance();
|
| 3587 |
-
}
|
| 3588 |
-
/**
|
| 3589 |
-
* @access private
|
| 3590 |
-
*/
|
| 3591 |
-
protected function merge() {
|
| 3592 |
-
foreach(func_get_args() as $nodes)
|
| 3593 |
-
foreach($nodes as $newNode )
|
| 3594 |
-
if (! $this->elementsContainsNode($newNode) )
|
| 3595 |
-
$this->elements[] = $newNode;
|
| 3596 |
-
}
|
| 3597 |
-
/**
|
| 3598 |
-
* @access private
|
| 3599 |
-
* TODO refactor to stackContainsNode
|
| 3600 |
-
*/
|
| 3601 |
-
protected function elementsContainsNode($nodeToCheck, $elementsStack = null) {
|
| 3602 |
-
$loop = ! is_null($elementsStack)
|
| 3603 |
-
? $elementsStack
|
| 3604 |
-
: $this->elements;
|
| 3605 |
-
foreach($loop as $node) {
|
| 3606 |
-
if ( $node->isSameNode( $nodeToCheck ) )
|
| 3607 |
-
return true;
|
| 3608 |
-
}
|
| 3609 |
-
return false;
|
| 3610 |
-
}
|
| 3611 |
-
/**
|
| 3612 |
-
* Enter description here...
|
| 3613 |
-
*
|
| 3614 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3615 |
-
*/
|
| 3616 |
-
public function parent($selector = null) {
|
| 3617 |
-
$stack = array();
|
| 3618 |
-
foreach($this->elements as $node )
|
| 3619 |
-
if ( $node->parentNode && ! $this->elementsContainsNode($node->parentNode, $stack) )
|
| 3620 |
-
$stack[] = $node->parentNode;
|
| 3621 |
-
$this->elementsBackup = $this->elements;
|
| 3622 |
-
$this->elements = $stack;
|
| 3623 |
-
if ( $selector )
|
| 3624 |
-
$this->filter($selector, true);
|
| 3625 |
-
return $this->newInstance();
|
| 3626 |
-
}
|
| 3627 |
-
/**
|
| 3628 |
-
* Enter description here...
|
| 3629 |
-
*
|
| 3630 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3631 |
-
*/
|
| 3632 |
-
public function parents($selector = null) {
|
| 3633 |
-
$stack = array();
|
| 3634 |
-
if (! $this->elements )
|
| 3635 |
-
$this->debug('parents() - stack empty');
|
| 3636 |
-
foreach($this->elements as $node) {
|
| 3637 |
-
$test = $node;
|
| 3638 |
-
while( $test->parentNode) {
|
| 3639 |
-
$test = $test->parentNode;
|
| 3640 |
-
if ($this->isRoot($test))
|
| 3641 |
-
break;
|
| 3642 |
-
if (! $this->elementsContainsNode($test, $stack)) {
|
| 3643 |
-
$stack[] = $test;
|
| 3644 |
-
continue;
|
| 3645 |
-
}
|
| 3646 |
-
}
|
| 3647 |
-
}
|
| 3648 |
-
$this->elementsBackup = $this->elements;
|
| 3649 |
-
$this->elements = $stack;
|
| 3650 |
-
if ( $selector )
|
| 3651 |
-
$this->filter($selector, true);
|
| 3652 |
-
return $this->newInstance();
|
| 3653 |
-
}
|
| 3654 |
-
/**
|
| 3655 |
-
* Internal stack iterator.
|
| 3656 |
-
*
|
| 3657 |
-
* @access private
|
| 3658 |
-
*/
|
| 3659 |
-
public function stack($nodeTypes = null) {
|
| 3660 |
-
if (!isset($nodeTypes))
|
| 3661 |
-
return $this->elements;
|
| 3662 |
-
if (!is_array($nodeTypes))
|
| 3663 |
-
$nodeTypes = array($nodeTypes);
|
| 3664 |
-
$return = array();
|
| 3665 |
-
foreach($this->elements as $node) {
|
| 3666 |
-
if (in_array($node->nodeType, $nodeTypes))
|
| 3667 |
-
$return[] = $node;
|
| 3668 |
-
}
|
| 3669 |
-
return $return;
|
| 3670 |
-
}
|
| 3671 |
-
// TODO phpdoc; $oldAttr is result of hasAttribute, before any changes
|
| 3672 |
-
protected function attrEvents($attr, $oldAttr, $oldValue, $node) {
|
| 3673 |
-
// skip events for XML documents
|
| 3674 |
-
if (! $this->isXHTML() && ! $this->isHTML())
|
| 3675 |
-
return;
|
| 3676 |
-
$event = null;
|
| 3677 |
-
// identify
|
| 3678 |
-
$isInputValue = $node->tagName == 'input'
|
| 3679 |
-
&& (
|
| 3680 |
-
in_array($node->getAttribute('type'),
|
| 3681 |
-
array('text', 'password', 'hidden'))
|
| 3682 |
-
|| !$node->getAttribute('type')
|
| 3683 |
-
);
|
| 3684 |
-
$isRadio = $node->tagName == 'input'
|
| 3685 |
-
&& $node->getAttribute('type') == 'radio';
|
| 3686 |
-
$isCheckbox = $node->tagName == 'input'
|
| 3687 |
-
&& $node->getAttribute('type') == 'checkbox';
|
| 3688 |
-
$isOption = $node->tagName == 'option';
|
| 3689 |
-
if ($isInputValue && $attr == 'value' && $oldValue != $node->getAttribute($attr)) {
|
| 3690 |
-
$event = new DOMEvent(array(
|
| 3691 |
-
'target' => $node,
|
| 3692 |
-
'type' => 'change'
|
| 3693 |
-
));
|
| 3694 |
-
} else if (($isRadio || $isCheckbox) && $attr == 'checked' && (
|
| 3695 |
-
// check
|
| 3696 |
-
(! $oldAttr && $node->hasAttribute($attr))
|
| 3697 |
-
// un-check
|
| 3698 |
-
|| (! $node->hasAttribute($attr) && $oldAttr)
|
| 3699 |
-
)) {
|
| 3700 |
-
$event = new DOMEvent(array(
|
| 3701 |
-
'target' => $node,
|
| 3702 |
-
'type' => 'change'
|
| 3703 |
-
));
|
| 3704 |
-
} else if ($isOption && $node->parentNode && $attr == 'selected' && (
|
| 3705 |
-
// select
|
| 3706 |
-
(! $oldAttr && $node->hasAttribute($attr))
|
| 3707 |
-
// un-select
|
| 3708 |
-
|| (! $node->hasAttribute($attr) && $oldAttr)
|
| 3709 |
-
)) {
|
| 3710 |
-
$event = new DOMEvent(array(
|
| 3711 |
-
'target' => $node->parentNode,
|
| 3712 |
-
'type' => 'change'
|
| 3713 |
-
));
|
| 3714 |
-
}
|
| 3715 |
-
if ($event) {
|
| 3716 |
-
phpQueryEvents::trigger($this->getDocumentID(),
|
| 3717 |
-
$event->type, array($event), $node
|
| 3718 |
-
);
|
| 3719 |
-
}
|
| 3720 |
-
}
|
| 3721 |
-
public function attr($attr = null, $value = null) {
|
| 3722 |
-
foreach($this->stack(1) as $node) {
|
| 3723 |
-
if (! is_null($value)) {
|
| 3724 |
-
$loop = $attr == '*'
|
| 3725 |
-
? $this->getNodeAttrs($node)
|
| 3726 |
-
: array($attr);
|
| 3727 |
-
foreach($loop as $a) {
|
| 3728 |
-
$oldValue = $node->getAttribute($a);
|
| 3729 |
-
$oldAttr = $node->hasAttribute($a);
|
| 3730 |
-
// TODO raises an error when charset other than UTF-8
|
| 3731 |
-
// while document's charset is also not UTF-8
|
| 3732 |
-
@$node->setAttribute($a, $value);
|
| 3733 |
-
$this->attrEvents($a, $oldAttr, $oldValue, $node);
|
| 3734 |
-
}
|
| 3735 |
-
} else if ($attr == '*') {
|
| 3736 |
-
// jQuery difference
|
| 3737 |
-
$return = array();
|
| 3738 |
-
foreach($node->attributes as $n => $v)
|
| 3739 |
-
$return[$n] = $v->value;
|
| 3740 |
-
return $return;
|
| 3741 |
-
} else
|
| 3742 |
-
return $node->hasAttribute($attr)
|
| 3743 |
-
? $node->getAttribute($attr)
|
| 3744 |
-
: null;
|
| 3745 |
-
}
|
| 3746 |
-
return is_null($value)
|
| 3747 |
-
? '' : $this;
|
| 3748 |
-
}
|
| 3749 |
-
/**
|
| 3750 |
-
* @access private
|
| 3751 |
-
*/
|
| 3752 |
-
protected function getNodeAttrs($node) {
|
| 3753 |
-
$return = array();
|
| 3754 |
-
foreach($node->attributes as $n => $o)
|
| 3755 |
-
$return[] = $n;
|
| 3756 |
-
return $return;
|
| 3757 |
-
}
|
| 3758 |
-
/**
|
| 3759 |
-
* Enter description here...
|
| 3760 |
-
*
|
| 3761 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3762 |
-
* @todo check CDATA ???
|
| 3763 |
-
*/
|
| 3764 |
-
public function attrPHP($attr, $code) {
|
| 3765 |
-
if (! is_null($code)) {
|
| 3766 |
-
$value = '<'.'?php '.$code.' ?'.'>';
|
| 3767 |
-
// TODO tempolary solution
|
| 3768 |
-
// http://code.google.com/p/phpquery/issues/detail?id=17
|
| 3769 |
-
// if (function_exists('mb_detect_encoding') && mb_detect_encoding($value) == 'ASCII')
|
| 3770 |
-
// $value = mb_convert_encoding($value, 'UTF-8', 'HTML-ENTITIES');
|
| 3771 |
-
}
|
| 3772 |
-
foreach($this->stack(1) as $node) {
|
| 3773 |
-
if (! is_null($code)) {
|
| 3774 |
-
// $attrNode = $this->DOM->createAttribute($attr);
|
| 3775 |
-
$node->setAttribute($attr, $value);
|
| 3776 |
-
// $attrNode->value = $value;
|
| 3777 |
-
// $node->appendChild($attrNode);
|
| 3778 |
-
} else if ( $attr == '*') {
|
| 3779 |
-
// jQuery diff
|
| 3780 |
-
$return = array();
|
| 3781 |
-
foreach($node->attributes as $n => $v)
|
| 3782 |
-
$return[$n] = $v->value;
|
| 3783 |
-
return $return;
|
| 3784 |
-
} else
|
| 3785 |
-
return $node->getAttribute($attr);
|
| 3786 |
-
}
|
| 3787 |
-
return $this;
|
| 3788 |
-
}
|
| 3789 |
-
/**
|
| 3790 |
-
* Enter description here...
|
| 3791 |
-
*
|
| 3792 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3793 |
-
*/
|
| 3794 |
-
public function removeAttr($attr) {
|
| 3795 |
-
foreach($this->stack(1) as $node) {
|
| 3796 |
-
$loop = $attr == '*'
|
| 3797 |
-
? $this->getNodeAttrs($node)
|
| 3798 |
-
: array($attr);
|
| 3799 |
-
foreach($loop as $a) {
|
| 3800 |
-
$oldValue = $node->getAttribute($a);
|
| 3801 |
-
$node->removeAttribute($a);
|
| 3802 |
-
$this->attrEvents($a, $oldValue, null, $node);
|
| 3803 |
-
}
|
| 3804 |
-
}
|
| 3805 |
-
return $this;
|
| 3806 |
-
}
|
| 3807 |
-
/**
|
| 3808 |
-
* Return form element value.
|
| 3809 |
-
*
|
| 3810 |
-
* @return String Fields value.
|
| 3811 |
-
*/
|
| 3812 |
-
public function val($val = null) {
|
| 3813 |
-
if (! isset($val)) {
|
| 3814 |
-
if ($this->eq(0)->is('select')) {
|
| 3815 |
-
$selected = $this->eq(0)->find('option[selected=selected]');
|
| 3816 |
-
if ($selected->is('[value]'))
|
| 3817 |
-
return $selected->attr('value');
|
| 3818 |
-
else
|
| 3819 |
-
return $selected->text();
|
| 3820 |
-
} else if ($this->eq(0)->is('textarea'))
|
| 3821 |
-
return $this->eq(0)->markup();
|
| 3822 |
-
else
|
| 3823 |
-
return $this->eq(0)->attr('value');
|
| 3824 |
-
} else {
|
| 3825 |
-
$_val = null;
|
| 3826 |
-
foreach($this->stack(1) as $node) {
|
| 3827 |
-
$node = pq($node, $this->getDocumentID());
|
| 3828 |
-
if (is_array($val) && in_array($node->attr('type'), array('checkbox', 'radio'))) {
|
| 3829 |
-
$isChecked = in_array($node->attr('value'), $val)
|
| 3830 |
-
|| in_array($node->attr('name'), $val);
|
| 3831 |
-
if ($isChecked)
|
| 3832 |
-
$node->attr('checked', 'checked');
|
| 3833 |
-
else
|
| 3834 |
-
$node->removeAttr('checked');
|
| 3835 |
-
} else if ($node->get(0)->tagName == 'select') {
|
| 3836 |
-
if (! isset($_val)) {
|
| 3837 |
-
$_val = array();
|
| 3838 |
-
if (! is_array($val))
|
| 3839 |
-
$_val = array((string)$val);
|
| 3840 |
-
else
|
| 3841 |
-
foreach($val as $v)
|
| 3842 |
-
$_val[] = $v;
|
| 3843 |
-
}
|
| 3844 |
-
foreach($node['option']->stack(1) as $option) {
|
| 3845 |
-
$option = pq($option, $this->getDocumentID());
|
| 3846 |
-
$selected = false;
|
| 3847 |
-
// XXX: workaround for string comparsion, see issue #96
|
| 3848 |
-
// http://code.google.com/p/phpquery/issues/detail?id=96
|
| 3849 |
-
$selected = is_null($option->attr('value'))
|
| 3850 |
-
? in_array($option->markup(), $_val)
|
| 3851 |
-
: in_array($option->attr('value'), $_val);
|
| 3852 |
-
// $optionValue = $option->attr('value');
|
| 3853 |
-
// $optionText = $option->text();
|
| 3854 |
-
// $optionTextLenght = mb_strlen($optionText);
|
| 3855 |
-
// foreach($_val as $v)
|
| 3856 |
-
// if ($optionValue == $v)
|
| 3857 |
-
// $selected = true;
|
| 3858 |
-
// else if ($optionText == $v && $optionTextLenght == mb_strlen($v))
|
| 3859 |
-
// $selected = true;
|
| 3860 |
-
if ($selected)
|
| 3861 |
-
$option->attr('selected', 'selected');
|
| 3862 |
-
else
|
| 3863 |
-
$option->removeAttr('selected');
|
| 3864 |
-
}
|
| 3865 |
-
} else if ($node->get(0)->tagName == 'textarea')
|
| 3866 |
-
$node->markup($val);
|
| 3867 |
-
else
|
| 3868 |
-
$node->attr('value', $val);
|
| 3869 |
-
}
|
| 3870 |
-
}
|
| 3871 |
-
return $this;
|
| 3872 |
-
}
|
| 3873 |
-
/**
|
| 3874 |
-
* Enter description here...
|
| 3875 |
-
*
|
| 3876 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3877 |
-
*/
|
| 3878 |
-
public function andSelf() {
|
| 3879 |
-
if ( $this->previous )
|
| 3880 |
-
$this->elements = array_merge($this->elements, $this->previous->elements);
|
| 3881 |
-
return $this;
|
| 3882 |
-
}
|
| 3883 |
-
/**
|
| 3884 |
-
* Enter description here...
|
| 3885 |
-
*
|
| 3886 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3887 |
-
*/
|
| 3888 |
-
public function addClass( $className) {
|
| 3889 |
-
if (! $className)
|
| 3890 |
-
return $this;
|
| 3891 |
-
foreach($this->stack(1) as $node) {
|
| 3892 |
-
if (! $this->is(".$className", $node))
|
| 3893 |
-
$node->setAttribute(
|
| 3894 |
-
'class',
|
| 3895 |
-
trim($node->getAttribute('class').' '.$className)
|
| 3896 |
-
);
|
| 3897 |
-
}
|
| 3898 |
-
return $this;
|
| 3899 |
-
}
|
| 3900 |
-
/**
|
| 3901 |
-
* Enter description here...
|
| 3902 |
-
*
|
| 3903 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3904 |
-
*/
|
| 3905 |
-
public function addClassPHP( $className) {
|
| 3906 |
-
foreach($this->stack(1) as $node) {
|
| 3907 |
-
$classes = $node->getAttribute('class');
|
| 3908 |
-
$newValue = $classes
|
| 3909 |
-
? $classes.' <'.'?php '.$className.' ?'.'>'
|
| 3910 |
-
: '<'.'?php '.$className.' ?'.'>';
|
| 3911 |
-
$node->setAttribute('class', $newValue);
|
| 3912 |
-
}
|
| 3913 |
-
return $this;
|
| 3914 |
-
}
|
| 3915 |
-
/**
|
| 3916 |
-
* Enter description here...
|
| 3917 |
-
*
|
| 3918 |
-
* @param string $className
|
| 3919 |
-
* @return bool
|
| 3920 |
-
*/
|
| 3921 |
-
public function hasClass($className) {
|
| 3922 |
-
foreach($this->stack(1) as $node) {
|
| 3923 |
-
if ( $this->is(".$className", $node))
|
| 3924 |
-
return true;
|
| 3925 |
-
}
|
| 3926 |
-
return false;
|
| 3927 |
-
}
|
| 3928 |
-
/**
|
| 3929 |
-
* Enter description here...
|
| 3930 |
-
*
|
| 3931 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3932 |
-
*/
|
| 3933 |
-
public function removeClass($className) {
|
| 3934 |
-
foreach($this->stack(1) as $node) {
|
| 3935 |
-
$classes = explode( ' ', $node->getAttribute('class'));
|
| 3936 |
-
if ( in_array($className, $classes)) {
|
| 3937 |
-
$classes = array_diff($classes, array($className));
|
| 3938 |
-
if ( $classes )
|
| 3939 |
-
$node->setAttribute('class', implode(' ', $classes));
|
| 3940 |
-
else
|
| 3941 |
-
$node->removeAttribute('class');
|
| 3942 |
-
}
|
| 3943 |
-
}
|
| 3944 |
-
return $this;
|
| 3945 |
-
}
|
| 3946 |
-
/**
|
| 3947 |
-
* Enter description here...
|
| 3948 |
-
*
|
| 3949 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3950 |
-
*/
|
| 3951 |
-
public function toggleClass($className) {
|
| 3952 |
-
foreach($this->stack(1) as $node) {
|
| 3953 |
-
if ( $this->is( $node, '.'.$className ))
|
| 3954 |
-
$this->removeClass($className);
|
| 3955 |
-
else
|
| 3956 |
-
$this->addClass($className);
|
| 3957 |
-
}
|
| 3958 |
-
return $this;
|
| 3959 |
-
}
|
| 3960 |
-
/**
|
| 3961 |
-
* Proper name without underscore (just ->empty()) also works.
|
| 3962 |
-
*
|
| 3963 |
-
* Removes all child nodes from the set of matched elements.
|
| 3964 |
-
*
|
| 3965 |
-
* Example:
|
| 3966 |
-
* pq("p")._empty()
|
| 3967 |
-
*
|
| 3968 |
-
* HTML:
|
| 3969 |
-
* <p>Hello, <span>Person</span> <a href="#">and person</a></p>
|
| 3970 |
-
*
|
| 3971 |
-
* Result:
|
| 3972 |
-
* [ <p></p> ]
|
| 3973 |
-
*
|
| 3974 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3975 |
-
* @access private
|
| 3976 |
-
*/
|
| 3977 |
-
public function _empty() {
|
| 3978 |
-
foreach($this->stack(1) as $node) {
|
| 3979 |
-
// thx to 'dave at dgx dot cz'
|
| 3980 |
-
$node->nodeValue = '';
|
| 3981 |
-
}
|
| 3982 |
-
return $this;
|
| 3983 |
-
}
|
| 3984 |
-
/**
|
| 3985 |
-
* Enter description here...
|
| 3986 |
-
*
|
| 3987 |
-
* @param array|string $callback Expects $node as first param, $index as second
|
| 3988 |
-
* @param array $scope External variables passed to callback. Use compact('varName1', 'varName2'...) and extract($scope)
|
| 3989 |
-
* @param array $arg1 Will ba passed as third and futher args to callback.
|
| 3990 |
-
* @param array $arg2 Will ba passed as fourth and futher args to callback, and so on...
|
| 3991 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3992 |
-
*/
|
| 3993 |
-
public function each($callback, $param1 = null, $param2 = null, $param3 = null) {
|
| 3994 |
-
$paramStructure = null;
|
| 3995 |
-
if (func_num_args() > 1) {
|
| 3996 |
-
$paramStructure = func_get_args();
|
| 3997 |
-
$paramStructure = array_slice($paramStructure, 1);
|
| 3998 |
-
}
|
| 3999 |
-
foreach($this->elements as $v)
|
| 4000 |
-
phpQuery::callbackRun($callback, array($v), $paramStructure);
|
| 4001 |
-
return $this;
|
| 4002 |
-
}
|
| 4003 |
-
/**
|
| 4004 |
-
* Run callback on actual object.
|
| 4005 |
-
*
|
| 4006 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4007 |
-
*/
|
| 4008 |
-
public function callback($callback, $param1 = null, $param2 = null, $param3 = null) {
|
| 4009 |
-
$params = func_get_args();
|
| 4010 |
-
$params[0] = $this;
|
| 4011 |
-
phpQuery::callbackRun($callback, $params);
|
| 4012 |
-
return $this;
|
| 4013 |
-
}
|
| 4014 |
-
/**
|
| 4015 |
-
* Enter description here...
|
| 4016 |
-
*
|
| 4017 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4018 |
-
* @todo add $scope and $args as in each() ???
|
| 4019 |
-
*/
|
| 4020 |
-
public function map($callback, $param1 = null, $param2 = null, $param3 = null) {
|
| 4021 |
-
// $stack = array();
|
| 4022 |
-
//// foreach($this->newInstance() as $node) {
|
| 4023 |
-
// foreach($this->newInstance() as $node) {
|
| 4024 |
-
// $result = call_user_func($callback, $node);
|
| 4025 |
-
// if ($result)
|
| 4026 |
-
// $stack[] = $result;
|
| 4027 |
-
// }
|
| 4028 |
-
$params = func_get_args();
|
| 4029 |
-
array_unshift($params, $this->elements);
|
| 4030 |
-
return $this->newInstance(
|
| 4031 |
-
call_user_func_array(array('phpQuery', 'map'), $params)
|
| 4032 |
-
// phpQuery::map($this->elements, $callback)
|
| 4033 |
-
);
|
| 4034 |
-
}
|
| 4035 |
-
/**
|
| 4036 |
-
* Enter description here...
|
| 4037 |
-
*
|
| 4038 |
-
* @param <type> $key
|
| 4039 |
-
* @param <type> $value
|
| 4040 |
-
*/
|
| 4041 |
-
public function data($key, $value = null) {
|
| 4042 |
-
if (! isset($value)) {
|
| 4043 |
-
// TODO? implement specific jQuery behavior od returning parent values
|
| 4044 |
-
// is child which we look up doesn't exist
|
| 4045 |
-
return phpQuery::data($this->get(0), $key, $value, $this->getDocumentID());
|
| 4046 |
-
} else {
|
| 4047 |
-
foreach($this as $node)
|
| 4048 |
-
phpQuery::data($node, $key, $value, $this->getDocumentID());
|
| 4049 |
-
return $this;
|
| 4050 |
-
}
|
| 4051 |
-
}
|
| 4052 |
-
/**
|
| 4053 |
-
* Enter description here...
|
| 4054 |
-
*
|
| 4055 |
-
* @param <type> $key
|
| 4056 |
-
*/
|
| 4057 |
-
public function removeData($key) {
|
| 4058 |
-
foreach($this as $node)
|
| 4059 |
-
phpQuery::removeData($node, $key, $this->getDocumentID());
|
| 4060 |
-
return $this;
|
| 4061 |
-
}
|
| 4062 |
-
// INTERFACE IMPLEMENTATIONS
|
| 4063 |
-
|
| 4064 |
-
// ITERATOR INTERFACE
|
| 4065 |
-
/**
|
| 4066 |
-
* @access private
|
| 4067 |
-
*/
|
| 4068 |
-
public function rewind(){
|
| 4069 |
-
$this->debug('iterating foreach');
|
| 4070 |
-
// phpQuery::selectDocument($this->getDocumentID());
|
| 4071 |
-
$this->elementsBackup = $this->elements;
|
| 4072 |
-
$this->elementsInterator = $this->elements;
|
| 4073 |
-
$this->valid = isset( $this->elements[0] )
|
| 4074 |
-
? 1 : 0;
|
| 4075 |
-
// $this->elements = $this->valid
|
| 4076 |
-
// ? array($this->elements[0])
|
| 4077 |
-
// : array();
|
| 4078 |
-
$this->current = 0;
|
| 4079 |
-
}
|
| 4080 |
-
/**
|
| 4081 |
-
* @access private
|
| 4082 |
-
*/
|
| 4083 |
-
public function current(){
|
| 4084 |
-
return $this->elementsInterator[ $this->current ];
|
| 4085 |
-
}
|
| 4086 |
-
/**
|
| 4087 |
-
* @access private
|
| 4088 |
-
*/
|
| 4089 |
-
public function key(){
|
| 4090 |
-
return $this->current;
|
| 4091 |
-
}
|
| 4092 |
-
/**
|
| 4093 |
-
* Double-function method.
|
| 4094 |
-
*
|
| 4095 |
-
* First: main iterator interface method.
|
| 4096 |
-
* Second: Returning next sibling, alias for _next().
|
| 4097 |
-
*
|
| 4098 |
-
* Proper functionality is choosed automagicaly.
|
| 4099 |
-
*
|
| 4100 |
-
* @see phpQueryObject::_next()
|
| 4101 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4102 |
-
*/
|
| 4103 |
-
public function next($cssSelector = null){
|
| 4104 |
-
// if ($cssSelector || $this->valid)
|
| 4105 |
-
// return $this->_next($cssSelector);
|
| 4106 |
-
$this->valid = isset( $this->elementsInterator[ $this->current+1 ] )
|
| 4107 |
-
? true
|
| 4108 |
-
: false;
|
| 4109 |
-
if (! $this->valid && $this->elementsInterator) {
|
| 4110 |
-
$this->elementsInterator = null;
|
| 4111 |
-
} else if ($this->valid) {
|
| 4112 |
-
$this->current++;
|
| 4113 |
-
} else {
|
| 4114 |
-
return $this->_next($cssSelector);
|
| 4115 |
-
}
|
| 4116 |
-
}
|
| 4117 |
-
/**
|
| 4118 |
-
* @access private
|
| 4119 |
-
*/
|
| 4120 |
-
public function valid(){
|
| 4121 |
-
return $this->valid;
|
| 4122 |
-
}
|
| 4123 |
-
// ITERATOR INTERFACE END
|
| 4124 |
-
// ARRAYACCESS INTERFACE
|
| 4125 |
-
/**
|
| 4126 |
-
* @access private
|
| 4127 |
-
*/
|
| 4128 |
-
public function offsetExists($offset) {
|
| 4129 |
-
return $this->find($offset)->size() > 0;
|
| 4130 |
-
}
|
| 4131 |
-
/**
|
| 4132 |
-
* @access private
|
| 4133 |
-
*/
|
| 4134 |
-
public function offsetGet($offset) {
|
| 4135 |
-
return $this->find($offset);
|
| 4136 |
-
}
|
| 4137 |
-
/**
|
| 4138 |
-
* @access private
|
| 4139 |
-
*/
|
| 4140 |
-
public function offsetSet($offset, $value) {
|
| 4141 |
-
// $this->find($offset)->replaceWith($value);
|
| 4142 |
-
$this->find($offset)->html($value);
|
| 4143 |
-
}
|
| 4144 |
-
/**
|
| 4145 |
-
* @access private
|
| 4146 |
-
*/
|
| 4147 |
-
public function offsetUnset($offset) {
|
| 4148 |
-
// empty
|
| 4149 |
-
throw new Exception("Can't do unset, use array interface only for calling queries and replacing HTML.");
|
| 4150 |
-
}
|
| 4151 |
-
// ARRAYACCESS INTERFACE END
|
| 4152 |
-
/**
|
| 4153 |
-
* Returns node's XPath.
|
| 4154 |
-
*
|
| 4155 |
-
* @param unknown_type $oneNode
|
| 4156 |
-
* @return string
|
| 4157 |
-
* @TODO use native getNodePath is avaible
|
| 4158 |
-
* @access private
|
| 4159 |
-
*/
|
| 4160 |
-
protected function getNodeXpath($oneNode = null, $namespace = null) {
|
| 4161 |
-
$return = array();
|
| 4162 |
-
$loop = $oneNode
|
| 4163 |
-
? array($oneNode)
|
| 4164 |
-
: $this->elements;
|
| 4165 |
-
// if ($namespace)
|
| 4166 |
-
// $namespace .= ':';
|
| 4167 |
-
foreach($loop as $node) {
|
| 4168 |
-
if ($node instanceof DOMDOCUMENT) {
|
| 4169 |
-
$return[] = '';
|
| 4170 |
-
continue;
|
| 4171 |
-
}
|
| 4172 |
-
$xpath = array();
|
| 4173 |
-
while(! ($node instanceof DOMDOCUMENT)) {
|
| 4174 |
-
$i = 1;
|
| 4175 |
-
$sibling = $node;
|
| 4176 |
-
while($sibling->previousSibling) {
|
| 4177 |
-
$sibling = $sibling->previousSibling;
|
| 4178 |
-
$isElement = $sibling instanceof DOMELEMENT;
|
| 4179 |
-
if ($isElement && $sibling->tagName == $node->tagName)
|
| 4180 |
-
$i++;
|
| 4181 |
-
}
|
| 4182 |
-
$xpath[] = $this->isXML()
|
| 4183 |
-
? "*[local-name()='{$node->tagName}'][{$i}]"
|
| 4184 |
-
: "{$node->tagName}[{$i}]";
|
| 4185 |
-
$node = $node->parentNode;
|
| 4186 |
-
}
|
| 4187 |
-
$xpath = join('/', array_reverse($xpath));
|
| 4188 |
-
$return[] = '/'.$xpath;
|
| 4189 |
-
}
|
| 4190 |
-
return $oneNode
|
| 4191 |
-
? $return[0]
|
| 4192 |
-
: $return;
|
| 4193 |
-
}
|
| 4194 |
-
// HELPERS
|
| 4195 |
-
public function whois($oneNode = null) {
|
| 4196 |
-
$return = array();
|
| 4197 |
-
$loop = $oneNode
|
| 4198 |
-
? array( $oneNode )
|
| 4199 |
-
: $this->elements;
|
| 4200 |
-
foreach($loop as $node) {
|
| 4201 |
-
if (isset($node->tagName)) {
|
| 4202 |
-
$tag = in_array($node->tagName, array('php', 'js'))
|
| 4203 |
-
? strtoupper($node->tagName)
|
| 4204 |
-
: $node->tagName;
|
| 4205 |
-
$return[] = $tag
|
| 4206 |
-
.($node->getAttribute('id')
|
| 4207 |
-
? '#'.$node->getAttribute('id'):'')
|
| 4208 |
-
.($node->getAttribute('class')
|
| 4209 |
-
? '.'.join('.', split(' ', $node->getAttribute('class'))):'')
|
| 4210 |
-
.($node->getAttribute('name')
|
| 4211 |
-
? '[name="'.$node->getAttribute('name').'"]':'')
|
| 4212 |
-
.($node->getAttribute('value') && strpos($node->getAttribute('value'), '<'.'?php') === false
|
| 4213 |
-
? '[value="'.substr(str_replace("\n", '', $node->getAttribute('value')), 0, 15).'"]':'')
|
| 4214 |
-
.($node->getAttribute('value') && strpos($node->getAttribute('value'), '<'.'?php') !== false
|
| 4215 |
-
? '[value=PHP]':'')
|
| 4216 |
-
.($node->getAttribute('selected')
|
| 4217 |
-
? '[selected]':'')
|
| 4218 |
-
.($node->getAttribute('checked')
|
| 4219 |
-
? '[checked]':'')
|
| 4220 |
-
;
|
| 4221 |
-
} else if ($node instanceof DOMTEXT) {
|
| 4222 |
-
if (trim($node->textContent))
|
| 4223 |
-
$return[] = 'Text:'.substr(str_replace("\n", ' ', $node->textContent), 0, 15);
|
| 4224 |
-
} else {
|
| 4225 |
-
|
| 4226 |
-
}
|
| 4227 |
-
}
|
| 4228 |
-
return $oneNode && isset($return[0])
|
| 4229 |
-
? $return[0]
|
| 4230 |
-
: $return;
|
| 4231 |
-
}
|
| 4232 |
-
/**
|
| 4233 |
-
* Dump htmlOuter and preserve chain. Usefull for debugging.
|
| 4234 |
-
*
|
| 4235 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4236 |
-
*
|
| 4237 |
-
*/
|
| 4238 |
-
public function dump() {
|
| 4239 |
-
print 'DUMP #'.(phpQuery::$dumpCount++).' ';
|
| 4240 |
-
$debug = phpQuery::$debug;
|
| 4241 |
-
phpQuery::$debug = false;
|
| 4242 |
-
// print __FILE__.':'.__LINE__."\n";
|
| 4243 |
-
var_dump($this->htmlOuter());
|
| 4244 |
-
return $this;
|
| 4245 |
-
}
|
| 4246 |
-
public function dumpWhois() {
|
| 4247 |
-
print 'DUMP #'.(phpQuery::$dumpCount++).' ';
|
| 4248 |
-
$debug = phpQuery::$debug;
|
| 4249 |
-
phpQuery::$debug = false;
|
| 4250 |
-
// print __FILE__.':'.__LINE__."\n";
|
| 4251 |
-
var_dump('whois', $this->whois());
|
| 4252 |
-
phpQuery::$debug = $debug;
|
| 4253 |
-
return $this;
|
| 4254 |
-
}
|
| 4255 |
-
public function dumpLength() {
|
| 4256 |
-
print 'DUMP #'.(phpQuery::$dumpCount++).' ';
|
| 4257 |
-
$debug = phpQuery::$debug;
|
| 4258 |
-
phpQuery::$debug = false;
|
| 4259 |
-
// print __FILE__.':'.__LINE__."\n";
|
| 4260 |
-
var_dump('length', $this->length());
|
| 4261 |
-
phpQuery::$debug = $debug;
|
| 4262 |
-
return $this;
|
| 4263 |
-
}
|
| 4264 |
-
public function dumpTree($html = true, $title = true) {
|
| 4265 |
-
$output = $title
|
| 4266 |
-
? 'DUMP #'.(phpQuery::$dumpCount++)." \n" : '';
|
| 4267 |
-
$debug = phpQuery::$debug;
|
| 4268 |
-
phpQuery::$debug = false;
|
| 4269 |
-
foreach($this->stack() as $node)
|
| 4270 |
-
$output .= $this->__dumpTree($node);
|
| 4271 |
-
phpQuery::$debug = $debug;
|
| 4272 |
-
print $html
|
| 4273 |
-
? nl2br(str_replace(' ', ' ', $output))
|
| 4274 |
-
: $output;
|
| 4275 |
-
return $this;
|
| 4276 |
-
}
|
| 4277 |
-
private function __dumpTree($node, $intend = 0) {
|
| 4278 |
-
$whois = $this->whois($node);
|
| 4279 |
-
$return = '';
|
| 4280 |
-
if ($whois)
|
| 4281 |
-
$return .= str_repeat(' - ', $intend).$whois."\n";
|
| 4282 |
-
if (isset($node->childNodes))
|
| 4283 |
-
foreach($node->childNodes as $chNode)
|
| 4284 |
-
$return .= $this->__dumpTree($chNode, $intend+1);
|
| 4285 |
-
return $return;
|
| 4286 |
-
}
|
| 4287 |
-
/**
|
| 4288 |
-
* Dump htmlOuter and stop script execution. Usefull for debugging.
|
| 4289 |
-
*
|
| 4290 |
-
*/
|
| 4291 |
-
public function dumpDie() {
|
| 4292 |
-
print __FILE__.':'.__LINE__;
|
| 4293 |
-
var_dump($this->htmlOuter());
|
| 4294 |
-
die();
|
| 4295 |
-
}
|
| 4296 |
-
}
|
| 4297 |
-
|
| 4298 |
-
|
| 4299 |
-
// -- Multibyte Compatibility functions ---------------------------------------
|
| 4300 |
-
// http://svn.iphonewebdev.com/lace/lib/mb_compat.php
|
| 4301 |
-
|
| 4302 |
-
/**
|
| 4303 |
-
* mb_internal_encoding()
|
| 4304 |
-
*
|
| 4305 |
-
* Included for mbstring pseudo-compatability.
|
| 4306 |
-
*/
|
| 4307 |
-
if (!function_exists('mb_internal_encoding'))
|
| 4308 |
-
{
|
| 4309 |
-
function mb_internal_encoding($enc) {return true; }
|
| 4310 |
-
}
|
| 4311 |
-
|
| 4312 |
-
/**
|
| 4313 |
-
* mb_regex_encoding()
|
| 4314 |
-
*
|
| 4315 |
-
* Included for mbstring pseudo-compatability.
|
| 4316 |
-
*/
|
| 4317 |
-
if (!function_exists('mb_regex_encoding'))
|
| 4318 |
-
{
|
| 4319 |
-
function mb_regex_encoding($enc) {return true; }
|
| 4320 |
-
}
|
| 4321 |
-
|
| 4322 |
-
/**
|
| 4323 |
-
* mb_strlen()
|
| 4324 |
-
*
|
| 4325 |
-
* Included for mbstring pseudo-compatability.
|
| 4326 |
-
*/
|
| 4327 |
-
if (!function_exists('mb_strlen'))
|
| 4328 |
-
{
|
| 4329 |
-
function mb_strlen($str)
|
| 4330 |
-
{
|
| 4331 |
-
return strlen($str);
|
| 4332 |
-
}
|
| 4333 |
-
}
|
| 4334 |
-
|
| 4335 |
-
/**
|
| 4336 |
-
* mb_strpos()
|
| 4337 |
-
*
|
| 4338 |
-
* Included for mbstring pseudo-compatability.
|
| 4339 |
-
*/
|
| 4340 |
-
if (!function_exists('mb_strpos'))
|
| 4341 |
-
{
|
| 4342 |
-
function mb_strpos($haystack, $needle, $offset=0)
|
| 4343 |
-
{
|
| 4344 |
-
return strpos($haystack, $needle, $offset);
|
| 4345 |
-
}
|
| 4346 |
-
}
|
| 4347 |
-
/**
|
| 4348 |
-
* mb_stripos()
|
| 4349 |
-
*
|
| 4350 |
-
* Included for mbstring pseudo-compatability.
|
| 4351 |
-
*/
|
| 4352 |
-
if (!function_exists('mb_stripos'))
|
| 4353 |
-
{
|
| 4354 |
-
function mb_stripos($haystack, $needle, $offset=0)
|
| 4355 |
-
{
|
| 4356 |
-
return stripos($haystack, $needle, $offset);
|
| 4357 |
-
}
|
| 4358 |
-
}
|
| 4359 |
-
|
| 4360 |
-
/**
|
| 4361 |
-
* mb_substr()
|
| 4362 |
-
*
|
| 4363 |
-
* Included for mbstring pseudo-compatability.
|
| 4364 |
-
*/
|
| 4365 |
-
if (!function_exists('mb_substr'))
|
| 4366 |
-
{
|
| 4367 |
-
function mb_substr($str, $start, $length=0)
|
| 4368 |
-
{
|
| 4369 |
-
return substr($str, $start, $length);
|
| 4370 |
-
}
|
| 4371 |
-
}
|
| 4372 |
-
|
| 4373 |
-
/**
|
| 4374 |
-
* mb_substr_count()
|
| 4375 |
-
*
|
| 4376 |
-
* Included for mbstring pseudo-compatability.
|
| 4377 |
-
*/
|
| 4378 |
-
if (!function_exists('mb_substr_count'))
|
| 4379 |
-
{
|
| 4380 |
-
function mb_substr_count($haystack, $needle)
|
| 4381 |
-
{
|
| 4382 |
-
return substr_count($haystack, $needle);
|
| 4383 |
-
}
|
| 4384 |
-
}
|
| 4385 |
-
|
| 4386 |
-
|
| 4387 |
-
/**
|
| 4388 |
-
* Static namespace for phpQuery functions.
|
| 4389 |
-
*
|
| 4390 |
-
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 4391 |
-
* @package phpQuery
|
| 4392 |
-
*/
|
| 4393 |
-
abstract class phpQuery {
|
| 4394 |
-
/**
|
| 4395 |
-
* XXX: Workaround for mbstring problems
|
| 4396 |
-
*
|
| 4397 |
-
* @var bool
|
| 4398 |
-
*/
|
| 4399 |
-
public static $mbstringSupport = true;
|
| 4400 |
-
public static $debug = false;
|
| 4401 |
-
public static $documents = array();
|
| 4402 |
-
public static $defaultDocumentID = null;
|
| 4403 |
-
// public static $defaultDoctype = 'html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"';
|
| 4404 |
-
/**
|
| 4405 |
-
* Applies only to HTML.
|
| 4406 |
-
*
|
| 4407 |
-
* @var unknown_type
|
| 4408 |
-
*/
|
| 4409 |
-
public static $defaultDoctype = '<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"
|
| 4410 |
-
"http://www.w3.org/TR/html4/loose.dtd">';
|
| 4411 |
-
public static $defaultCharset = 'UTF-8';
|
| 4412 |
-
/**
|
| 4413 |
-
* Static namespace for plugins.
|
| 4414 |
-
*
|
| 4415 |
-
* @var object
|
| 4416 |
-
*/
|
| 4417 |
-
public static $plugins = array();
|
| 4418 |
-
/**
|
| 4419 |
-
* List of loaded plugins.
|
| 4420 |
-
*
|
| 4421 |
-
* @var unknown_type
|
| 4422 |
-
*/
|
| 4423 |
-
public static $pluginsLoaded = array();
|
| 4424 |
-
public static $pluginsMethods = array();
|
| 4425 |
-
public static $pluginsStaticMethods = array();
|
| 4426 |
-
public static $extendMethods = array();
|
| 4427 |
-
/**
|
| 4428 |
-
* @TODO implement
|
| 4429 |
-
*/
|
| 4430 |
-
public static $extendStaticMethods = array();
|
| 4431 |
-
/**
|
| 4432 |
-
* Hosts allowed for AJAX connections.
|
| 4433 |
-
* Dot '.' means $_SERVER['HTTP_HOST'] (if any).
|
| 4434 |
-
*
|
| 4435 |
-
* @var array
|
| 4436 |
-
*/
|
| 4437 |
-
public static $ajaxAllowedHosts = array(
|
| 4438 |
-
'.'
|
| 4439 |
-
);
|
| 4440 |
-
/**
|
| 4441 |
-
* AJAX settings.
|
| 4442 |
-
*
|
| 4443 |
-
* @var array
|
| 4444 |
-
* XXX should it be static or not ?
|
| 4445 |
-
*/
|
| 4446 |
-
public static $ajaxSettings = array(
|
| 4447 |
-
'url' => '',//TODO
|
| 4448 |
-
'global' => true,
|
| 4449 |
-
'type' => "GET",
|
| 4450 |
-
'timeout' => null,
|
| 4451 |
-
'contentType' => "application/x-www-form-urlencoded",
|
| 4452 |
-
'processData' => true,
|
| 4453 |
-
// 'async' => true,
|
| 4454 |
-
'data' => null,
|
| 4455 |
-
'username' => null,
|
| 4456 |
-
'password' => null,
|
| 4457 |
-
'accepts' => array(
|
| 4458 |
-
'xml' => "application/xml, text/xml",
|
| 4459 |
-
'html' => "text/html",
|
| 4460 |
-
'script' => "text/javascript, application/javascript",
|
| 4461 |
-
'json' => "application/json, text/javascript",
|
| 4462 |
-
'text' => "text/plain",
|
| 4463 |
-
'_default' => "*/*"
|
| 4464 |
-
)
|
| 4465 |
-
);
|
| 4466 |
-
public static $lastModified = null;
|
| 4467 |
-
public static $active = 0;
|
| 4468 |
-
public static $dumpCount = 0;
|
| 4469 |
-
/**
|
| 4470 |
-
* Multi-purpose function.
|
| 4471 |
-
* Use pq() as shortcut.
|
| 4472 |
-
*
|
| 4473 |
-
* In below examples, $pq is any result of pq(); function.
|
| 4474 |
-
*
|
| 4475 |
-
* 1. Import markup into existing document (without any attaching):
|
| 4476 |
-
* - Import into selected document:
|
| 4477 |
-
* pq('<div/>') // DOESNT accept text nodes at beginning of input string !
|
| 4478 |
-
* - Import into document with ID from $pq->getDocumentID():
|
| 4479 |
-
* pq('<div/>', $pq->getDocumentID())
|
| 4480 |
-
* - Import into same document as DOMNode belongs to:
|
| 4481 |
-
* pq('<div/>', DOMNode)
|
| 4482 |
-
* - Import into document from phpQuery object:
|
| 4483 |
-
* pq('<div/>', $pq)
|
| 4484 |
-
*
|
| 4485 |
-
* 2. Run query:
|
| 4486 |
-
* - Run query on last selected document:
|
| 4487 |
-
* pq('div.myClass')
|
| 4488 |
-
* - Run query on document with ID from $pq->getDocumentID():
|
| 4489 |
-
* pq('div.myClass', $pq->getDocumentID())
|
| 4490 |
-
* - Run query on same document as DOMNode belongs to and use node(s)as root for query:
|
| 4491 |
-
* pq('div.myClass', DOMNode)
|
| 4492 |
-
* - Run query on document from phpQuery object
|
| 4493 |
-
* and use object's stack as root node(s) for query:
|
| 4494 |
-
* pq('div.myClass', $pq)
|
| 4495 |
-
*
|
| 4496 |
-
* @param string|DOMNode|DOMNodeList|array $arg1 HTML markup, CSS Selector, DOMNode or array of DOMNodes
|
| 4497 |
-
* @param string|phpQueryObject|DOMNode $context DOM ID from $pq->getDocumentID(), phpQuery object (determines also query root) or DOMNode (determines also query root)
|
| 4498 |
-
*
|
| 4499 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery|QueryTemplatesPhpQuery|false
|
| 4500 |
-
* phpQuery object or false in case of error.
|
| 4501 |
-
*/
|
| 4502 |
-
public static function pq($arg1, $context = null) {
|
| 4503 |
-
if ($arg1 instanceof DOMNODE && ! isset($context)) {
|
| 4504 |
-
foreach(phpQuery::$documents as $documentWrapper) {
|
| 4505 |
-
$compare = $arg1 instanceof DOMDocument
|
| 4506 |
-
? $arg1 : $arg1->ownerDocument;
|
| 4507 |
-
if ($documentWrapper->document->isSameNode($compare))
|
| 4508 |
-
$context = $documentWrapper->id;
|
| 4509 |
-
}
|
| 4510 |
-
}
|
| 4511 |
-
if (! $context) {
|
| 4512 |
-
$domId = self::$defaultDocumentID;
|
| 4513 |
-
if (! $domId)
|
| 4514 |
-
throw new Exception("Can't use last created DOM, because there isn't any. Use phpQuery::newDocument() first.");
|
| 4515 |
-
// } else if (is_object($context) && ($context instanceof PHPQUERY || is_subclass_of($context, 'phpQueryObject')))
|
| 4516 |
-
} else if (is_object($context) && $context instanceof phpQueryObject)
|
| 4517 |
-
$domId = $context->getDocumentID();
|
| 4518 |
-
else if ($context instanceof DOMDOCUMENT) {
|
| 4519 |
-
$domId = self::getDocumentID($context);
|
| 4520 |
-
if (! $domId) {
|
| 4521 |
-
//throw new Exception('Orphaned DOMDocument');
|
| 4522 |
-
$domId = self::newDocument($context)->getDocumentID();
|
| 4523 |
-
}
|
| 4524 |
-
} else if ($context instanceof DOMNODE) {
|
| 4525 |
-
$domId = self::getDocumentID($context);
|
| 4526 |
-
if (! $domId) {
|
| 4527 |
-
throw new Exception('Orphaned DOMNode');
|
| 4528 |
-
// $domId = self::newDocument($context->ownerDocument);
|
| 4529 |
-
}
|
| 4530 |
-
} else
|
| 4531 |
-
$domId = $context;
|
| 4532 |
-
if ($arg1 instanceof phpQueryObject) {
|
| 4533 |
-
// if (is_object($arg1) && (get_class($arg1) == 'phpQueryObject' || $arg1 instanceof PHPQUERY || is_subclass_of($arg1, 'phpQueryObject'))) {
|
| 4534 |
-
/**
|
| 4535 |
-
* Return $arg1 or import $arg1 stack if document differs:
|
| 4536 |
-
* pq(pq('<div/>'))
|
| 4537 |
-
*/
|
| 4538 |
-
if ($arg1->getDocumentID() == $domId)
|
| 4539 |
-
return $arg1;
|
| 4540 |
-
$class = get_class($arg1);
|
| 4541 |
-
// support inheritance by passing old object to overloaded constructor
|
| 4542 |
-
$phpQuery = $class != 'phpQuery'
|
| 4543 |
-
? new $class($arg1, $domId)
|
| 4544 |
-
: new phpQueryObject($domId);
|
| 4545 |
-
$phpQuery->elements = array();
|
| 4546 |
-
foreach($arg1->elements as $node)
|
| 4547 |
-
$phpQuery->elements[] = $phpQuery->document->importNode($node, true);
|
| 4548 |
-
return $phpQuery;
|
| 4549 |
-
} else if ($arg1 instanceof DOMNODE || (is_array($arg1) && isset($arg1[0]) && $arg1[0] instanceof DOMNODE)) {
|
| 4550 |
-
/*
|
| 4551 |
-
* Wrap DOM nodes with phpQuery object, import into document when needed:
|
| 4552 |
-
* pq(array($domNode1, $domNode2))
|
| 4553 |
-
*/
|
| 4554 |
-
$phpQuery = new phpQueryObject($domId);
|
| 4555 |
-
if (!($arg1 instanceof DOMNODELIST) && ! is_array($arg1))
|
| 4556 |
-
$arg1 = array($arg1);
|
| 4557 |
-
$phpQuery->elements = array();
|
| 4558 |
-
foreach($arg1 as $node) {
|
| 4559 |
-
$sameDocument = $node->ownerDocument instanceof DOMDOCUMENT
|
| 4560 |
-
&& ! $node->ownerDocument->isSameNode($phpQuery->document);
|
| 4561 |
-
$phpQuery->elements[] = $sameDocument
|
| 4562 |
-
? $phpQuery->document->importNode($node, true)
|
| 4563 |
-
: $node;
|
| 4564 |
-
}
|
| 4565 |
-
return $phpQuery;
|
| 4566 |
-
} else if (self::isMarkup($arg1)) {
|
| 4567 |
-
/**
|
| 4568 |
-
* Import HTML:
|
| 4569 |
-
* pq('<div/>')
|
| 4570 |
-
*/
|
| 4571 |
-
$phpQuery = new phpQueryObject($domId);
|
| 4572 |
-
return $phpQuery->newInstance(
|
| 4573 |
-
$phpQuery->documentWrapper->import($arg1)
|
| 4574 |
-
);
|
| 4575 |
-
} else {
|
| 4576 |
-
/**
|
| 4577 |
-
* Run CSS query:
|
| 4578 |
-
* pq('div.myClass')
|
| 4579 |
-
*/
|
| 4580 |
-
$phpQuery = new phpQueryObject($domId);
|
| 4581 |
-
// if ($context && ($context instanceof PHPQUERY || is_subclass_of($context, 'phpQueryObject')))
|
| 4582 |
-
if ($context && $context instanceof phpQueryObject)
|
| 4583 |
-
$phpQuery->elements = $context->elements;
|
| 4584 |
-
else if ($context && $context instanceof DOMNODELIST) {
|
| 4585 |
-
$phpQuery->elements = array();
|
| 4586 |
-
foreach($context as $node)
|
| 4587 |
-
$phpQuery->elements[] = $node;
|
| 4588 |
-
} else if ($context && $context instanceof DOMNODE)
|
| 4589 |
-
$phpQuery->elements = array($context);
|
| 4590 |
-
return $phpQuery->find($arg1);
|
| 4591 |
-
}
|
| 4592 |
-
}
|
| 4593 |
-
/**
|
| 4594 |
-
* Sets default document to $id. Document has to be loaded prior
|
| 4595 |
-
* to using this method.
|
| 4596 |
-
* $id can be retrived via getDocumentID() or getDocumentIDRef().
|
| 4597 |
-
*
|
| 4598 |
-
* @param unknown_type $id
|
| 4599 |
-
*/
|
| 4600 |
-
public static function selectDocument($id) {
|
| 4601 |
-
$id = self::getDocumentID($id);
|
| 4602 |
-
self::debug("Selecting document '$id' as default one");
|
| 4603 |
-
self::$defaultDocumentID = self::getDocumentID($id);
|
| 4604 |
-
}
|
| 4605 |
-
/**
|
| 4606 |
-
* Returns document with id $id or last used as phpQueryObject.
|
| 4607 |
-
* $id can be retrived via getDocumentID() or getDocumentIDRef().
|
| 4608 |
-
* Chainable.
|
| 4609 |
-
*
|
| 4610 |
-
* @see phpQuery::selectDocument()
|
| 4611 |
-
* @param unknown_type $id
|
| 4612 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4613 |
-
*/
|
| 4614 |
-
public static function getDocument($id = null) {
|
| 4615 |
-
if ($id)
|
| 4616 |
-
phpQuery::selectDocument($id);
|
| 4617 |
-
else
|
| 4618 |
-
$id = phpQuery::$defaultDocumentID;
|
| 4619 |
-
return new phpQueryObject($id);
|
| 4620 |
-
}
|
| 4621 |
-
/**
|
| 4622 |
-
* Creates new document from markup.
|
| 4623 |
-
* Chainable.
|
| 4624 |
-
*
|
| 4625 |
-
* @param unknown_type $markup
|
| 4626 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4627 |
-
*/
|
| 4628 |
-
public static function newDocument($markup = null, $contentType = null) {
|
| 4629 |
-
if (! $markup)
|
| 4630 |
-
$markup = '';
|
| 4631 |
-
$documentID = phpQuery::createDocumentWrapper($markup, $contentType);
|
| 4632 |
-
return new phpQueryObject($documentID);
|
| 4633 |
-
}
|
| 4634 |
-
/**
|
| 4635 |
-
* Creates new document from markup.
|
| 4636 |
-
* Chainable.
|
| 4637 |
-
*
|
| 4638 |
-
* @param unknown_type $markup
|
| 4639 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4640 |
-
*/
|
| 4641 |
-
public static function newDocumentHTML($markup = null, $charset = null) {
|
| 4642 |
-
$contentType = $charset
|
| 4643 |
-
? ";charset=$charset"
|
| 4644 |
-
: '';
|
| 4645 |
-
return self::newDocument($markup, "text/html{$contentType}");
|
| 4646 |
-
}
|
| 4647 |
-
/**
|
| 4648 |
-
* Creates new document from markup.
|
| 4649 |
-
* Chainable.
|
| 4650 |
-
*
|
| 4651 |
-
* @param unknown_type $markup
|
| 4652 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4653 |
-
*/
|
| 4654 |
-
public static function newDocumentXML($markup = null, $charset = null) {
|
| 4655 |
-
$contentType = $charset
|
| 4656 |
-
? ";charset=$charset"
|
| 4657 |
-
: '';
|
| 4658 |
-
return self::newDocument($markup, "text/xml{$contentType}");
|
| 4659 |
-
}
|
| 4660 |
-
/**
|
| 4661 |
-
* Creates new document from markup.
|
| 4662 |
-
* Chainable.
|
| 4663 |
-
*
|
| 4664 |
-
* @param unknown_type $markup
|
| 4665 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4666 |
-
*/
|
| 4667 |
-
public static function newDocumentXHTML($markup = null, $charset = null) {
|
| 4668 |
-
$contentType = $charset
|
| 4669 |
-
? ";charset=$charset"
|
| 4670 |
-
: '';
|
| 4671 |
-
return self::newDocument($markup, "application/xhtml+xml{$contentType}");
|
| 4672 |
-
}
|
| 4673 |
-
/**
|
| 4674 |
-
* Creates new document from markup.
|
| 4675 |
-
* Chainable.
|
| 4676 |
-
*
|
| 4677 |
-
* @param unknown_type $markup
|
| 4678 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4679 |
-
*/
|
| 4680 |
-
public static function newDocumentPHP($markup = null, $contentType = "text/html") {
|
| 4681 |
-
// TODO pass charset to phpToMarkup if possible (use DOMDocumentWrapper function)
|
| 4682 |
-
$markup = phpQuery::phpToMarkup($markup, self::$defaultCharset);
|
| 4683 |
-
return self::newDocument($markup, $contentType);
|
| 4684 |
-
}
|
| 4685 |
-
public static function phpToMarkup($php, $charset = 'utf-8') {
|
| 4686 |
-
$regexes = array(
|
| 4687 |
-
'@(<(?!\\?)(?:[^>]|\\?>)+\\w+\\s*=\\s*)(\')([^\']*)<'.'?php?(.*?)(?:\\?>)([^\']*)\'@s',
|
| 4688 |
-
'@(<(?!\\?)(?:[^>]|\\?>)+\\w+\\s*=\\s*)(")([^"]*)<'.'?php?(.*?)(?:\\?>)([^"]*)"@s',
|
| 4689 |
-
);
|
| 4690 |
-
foreach($regexes as $regex)
|
| 4691 |
-
while (preg_match($regex, $php, $matches)) {
|
| 4692 |
-
$php = preg_replace_callback(
|
| 4693 |
-
$regex,
|
| 4694 |
-
// create_function('$m, $charset = "'.$charset.'"',
|
| 4695 |
-
// 'return $m[1].$m[2]
|
| 4696 |
-
// .htmlspecialchars("<"."?php".$m[4]."?".">", ENT_QUOTES|ENT_NOQUOTES, $charset)
|
| 4697 |
-
// .$m[5].$m[2];'
|
| 4698 |
-
// ),
|
| 4699 |
-
array('phpQuery', '_phpToMarkupCallback'),
|
| 4700 |
-
$php
|
| 4701 |
-
);
|
| 4702 |
-
}
|
| 4703 |
-
$regex = '@(^|>[^<]*)+?(<\?php(.*?)(\?>))@s';
|
| 4704 |
-
//preg_match_all($regex, $php, $matches);
|
| 4705 |
-
//var_dump($matches);
|
| 4706 |
-
$php = preg_replace($regex, '\\1<php><!-- \\3 --></php>', $php);
|
| 4707 |
-
return $php;
|
| 4708 |
-
}
|
| 4709 |
-
public static function _phpToMarkupCallback($php, $charset = 'utf-8') {
|
| 4710 |
-
return $m[1].$m[2]
|
| 4711 |
-
.htmlspecialchars("<"."?php".$m[4]."?".">", ENT_QUOTES|ENT_NOQUOTES, $charset)
|
| 4712 |
-
.$m[5].$m[2];
|
| 4713 |
-
}
|
| 4714 |
-
public static function _markupToPHPCallback($m) {
|
| 4715 |
-
return "<"."?php ".htmlspecialchars_decode($m[1])." ?".">";
|
| 4716 |
-
}
|
| 4717 |
-
/**
|
| 4718 |
-
* Converts document markup containing PHP code generated by phpQuery::php()
|
| 4719 |
-
* into valid (executable) PHP code syntax.
|
| 4720 |
-
*
|
| 4721 |
-
* @param string|phpQueryObject $content
|
| 4722 |
-
* @return string PHP code.
|
| 4723 |
-
*/
|
| 4724 |
-
public static function markupToPHP($content) {
|
| 4725 |
-
if ($content instanceof phpQueryObject)
|
| 4726 |
-
$content = $content->markupOuter();
|
| 4727 |
-
/* <php>...</php> to <?php...? > */
|
| 4728 |
-
$content = preg_replace_callback(
|
| 4729 |
-
'@<php>\s*<!--(.*?)-->\s*</php>@s',
|
| 4730 |
-
// create_function('$m',
|
| 4731 |
-
// 'return "<'.'?php ".htmlspecialchars_decode($m[1])." ?'.'>";'
|
| 4732 |
-
// ),
|
| 4733 |
-
array('phpQuery', '_markupToPHPCallback'),
|
| 4734 |
-
$content
|
| 4735 |
-
);
|
| 4736 |
-
/* <node attr='< ?php ? >'> extra space added to save highlighters */
|
| 4737 |
-
$regexes = array(
|
| 4738 |
-
'@(<(?!\\?)(?:[^>]|\\?>)+\\w+\\s*=\\s*)(\')([^\']*)(?:<|%3C)\\?(?:php)?(.*?)(?:\\?(?:>|%3E))([^\']*)\'@s',
|
| 4739 |
-
'@(<(?!\\?)(?:[^>]|\\?>)+\\w+\\s*=\\s*)(")([^"]*)(?:<|%3C)\\?(?:php)?(.*?)(?:\\?(?:>|%3E))([^"]*)"@s',
|
| 4740 |
-
);
|
| 4741 |
-
foreach($regexes as $regex)
|
| 4742 |
-
while (preg_match($regex, $content))
|
| 4743 |
-
$content = preg_replace_callback(
|
| 4744 |
-
$regex,
|
| 4745 |
-
create_function('$m',
|
| 4746 |
-
'return $m[1].$m[2].$m[3]."<?php "
|
| 4747 |
-
.str_replace(
|
| 4748 |
-
array("%20", "%3E", "%09", " ", "	", "%7B", "%24", "%7D", "%22", "%5B", "%5D"),
|
| 4749 |
-
array(" ", ">", " ", "\n", " ", "{", "$", "}", \'"\', "[", "]"),
|
| 4750 |
-
htmlspecialchars_decode($m[4])
|
| 4751 |
-
)
|
| 4752 |
-
." ?>".$m[5].$m[2];'
|
| 4753 |
-
),
|
| 4754 |
-
$content
|
| 4755 |
-
);
|
| 4756 |
-
return $content;
|
| 4757 |
-
}
|
| 4758 |
-
/**
|
| 4759 |
-
* Creates new document from file $file.
|
| 4760 |
-
* Chainable.
|
| 4761 |
-
*
|
| 4762 |
-
* @param string $file URLs allowed. See File wrapper page at php.net for more supported sources.
|
| 4763 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4764 |
-
*/
|
| 4765 |
-
public static function newDocumentFile($file, $contentType = null) {
|
| 4766 |
-
$documentID = self::createDocumentWrapper(
|
| 4767 |
-
file_get_contents($file), $contentType
|
| 4768 |
-
);
|
| 4769 |
-
return new phpQueryObject($documentID);
|
| 4770 |
-
}
|
| 4771 |
-
/**
|
| 4772 |
-
* Creates new document from markup.
|
| 4773 |
-
* Chainable.
|
| 4774 |
-
*
|
| 4775 |
-
* @param unknown_type $markup
|
| 4776 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4777 |
-
*/
|
| 4778 |
-
public static function newDocumentFileHTML($file, $charset = null) {
|
| 4779 |
-
$contentType = $charset
|
| 4780 |
-
? ";charset=$charset"
|
| 4781 |
-
: '';
|
| 4782 |
-
return self::newDocumentFile($file, "text/html{$contentType}");
|
| 4783 |
-
}
|
| 4784 |
-
/**
|
| 4785 |
-
* Creates new document from markup.
|
| 4786 |
-
* Chainable.
|
| 4787 |
-
*
|
| 4788 |
-
* @param unknown_type $markup
|
| 4789 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4790 |
-
*/
|
| 4791 |
-
public static function newDocumentFileXML($file, $charset = null) {
|
| 4792 |
-
$contentType = $charset
|
| 4793 |
-
? ";charset=$charset"
|
| 4794 |
-
: '';
|
| 4795 |
-
return self::newDocumentFile($file, "text/xml{$contentType}");
|
| 4796 |
-
}
|
| 4797 |
-
/**
|
| 4798 |
-
* Creates new document from markup.
|
| 4799 |
-
* Chainable.
|
| 4800 |
-
*
|
| 4801 |
-
* @param unknown_type $markup
|
| 4802 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4803 |
-
*/
|
| 4804 |
-
public static function newDocumentFileXHTML($file, $charset = null) {
|
| 4805 |
-
$contentType = $charset
|
| 4806 |
-
? ";charset=$charset"
|
| 4807 |
-
: '';
|
| 4808 |
-
return self::newDocumentFile($file, "application/xhtml+xml{$contentType}");
|
| 4809 |
-
}
|
| 4810 |
-
/**
|
| 4811 |
-
* Creates new document from markup.
|
| 4812 |
-
* Chainable.
|
| 4813 |
-
*
|
| 4814 |
-
* @param unknown_type $markup
|
| 4815 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4816 |
-
*/
|
| 4817 |
-
public static function newDocumentFilePHP($file, $contentType = null) {
|
| 4818 |
-
return self::newDocumentPHP(file_get_contents($file), $contentType);
|
| 4819 |
-
}
|
| 4820 |
-
/**
|
| 4821 |
-
* Reuses existing DOMDocument object.
|
| 4822 |
-
* Chainable.
|
| 4823 |
-
*
|
| 4824 |
-
* @param $document DOMDocument
|
| 4825 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4826 |
-
* @TODO support DOMDocument
|
| 4827 |
-
*/
|
| 4828 |
-
public static function loadDocument($document) {
|
| 4829 |
-
// TODO
|
| 4830 |
-
die('TODO loadDocument');
|
| 4831 |
-
}
|
| 4832 |
-
/**
|
| 4833 |
-
* Enter description here...
|
| 4834 |
-
*
|
| 4835 |
-
* @param unknown_type $html
|
| 4836 |
-
* @param unknown_type $domId
|
| 4837 |
-
* @return unknown New DOM ID
|
| 4838 |
-
* @todo support PHP tags in input
|
| 4839 |
-
* @todo support passing DOMDocument object from self::loadDocument
|
| 4840 |
-
*/
|
| 4841 |
-
protected static function createDocumentWrapper($html, $contentType = null, $documentID = null) {
|
| 4842 |
-
if (function_exists('domxml_open_mem'))
|
| 4843 |
-
throw new Exception("Old PHP4 DOM XML extension detected. phpQuery won't work until this extension is enabled.");
|
| 4844 |
-
// $id = $documentID
|
| 4845 |
-
// ? $documentID
|
| 4846 |
-
// : md5(microtime());
|
| 4847 |
-
$document = null;
|
| 4848 |
-
if ($html instanceof DOMDOCUMENT) {
|
| 4849 |
-
if (self::getDocumentID($html)) {
|
| 4850 |
-
// document already exists in phpQuery::$documents, make a copy
|
| 4851 |
-
$document = clone $html;
|
| 4852 |
-
} else {
|
| 4853 |
-
// new document, add it to phpQuery::$documents
|
| 4854 |
-
$wrapper = new DOMDocumentWrapper($html, $contentType, $documentID);
|
| 4855 |
-
}
|
| 4856 |
-
} else {
|
| 4857 |
-
$wrapper = new DOMDocumentWrapper($html, $contentType, $documentID);
|
| 4858 |
-
}
|
| 4859 |
-
// $wrapper->id = $id;
|
| 4860 |
-
// bind document
|
| 4861 |
-
phpQuery::$documents[$wrapper->id] = $wrapper;
|
| 4862 |
-
// remember last loaded document
|
| 4863 |
-
phpQuery::selectDocument($wrapper->id);
|
| 4864 |
-
return $wrapper->id;
|
| 4865 |
-
}
|
| 4866 |
-
/**
|
| 4867 |
-
* Extend class namespace.
|
| 4868 |
-
*
|
| 4869 |
-
* @param string|array $target
|
| 4870 |
-
* @param array $source
|
| 4871 |
-
* @TODO support string $source
|
| 4872 |
-
* @return unknown_type
|
| 4873 |
-
*/
|
| 4874 |
-
public static function extend($target, $source) {
|
| 4875 |
-
switch($target) {
|
| 4876 |
-
case 'phpQueryObject':
|
| 4877 |
-
$targetRef = &self::$extendMethods;
|
| 4878 |
-
$targetRef2 = &self::$pluginsMethods;
|
| 4879 |
-
break;
|
| 4880 |
-
case 'phpQuery':
|
| 4881 |
-
$targetRef = &self::$extendStaticMethods;
|
| 4882 |
-
$targetRef2 = &self::$pluginsStaticMethods;
|
| 4883 |
-
break;
|
| 4884 |
-
default:
|
| 4885 |
-
throw new Exception("Unsupported \$target type");
|
| 4886 |
-
}
|
| 4887 |
-
if (is_string($source))
|
| 4888 |
-
$source = array($source => $source);
|
| 4889 |
-
foreach($source as $method => $callback) {
|
| 4890 |
-
if (isset($targetRef[$method])) {
|
| 4891 |
-
// throw new Exception
|
| 4892 |
-
self::debug("Duplicate method '{$method}', can\'t extend '{$target}'");
|
| 4893 |
-
continue;
|
| 4894 |
-
}
|
| 4895 |
-
if (isset($targetRef2[$method])) {
|
| 4896 |
-
// throw new Exception
|
| 4897 |
-
self::debug("Duplicate method '{$method}' from plugin '{$targetRef2[$method]}',"
|
| 4898 |
-
." can\'t extend '{$target}'");
|
| 4899 |
-
continue;
|
| 4900 |
-
}
|
| 4901 |
-
$targetRef[$method] = $callback;
|
| 4902 |
-
}
|
| 4903 |
-
return true;
|
| 4904 |
-
}
|
| 4905 |
-
/**
|
| 4906 |
-
* Extend phpQuery with $class from $file.
|
| 4907 |
-
*
|
| 4908 |
-
* @param string $class Extending class name. Real class name can be prepended phpQuery_.
|
| 4909 |
-
* @param string $file Filename to include. Defaults to "{$class}.php".
|
| 4910 |
-
*/
|
| 4911 |
-
public static function plugin($class, $file = null) {
|
| 4912 |
-
// TODO $class checked agains phpQuery_$class
|
| 4913 |
-
// if (strpos($class, 'phpQuery') === 0)
|
| 4914 |
-
// $class = substr($class, 8);
|
| 4915 |
-
if (in_array($class, self::$pluginsLoaded))
|
| 4916 |
-
return true;
|
| 4917 |
-
if (! $file)
|
| 4918 |
-
$file = $class.'.php';
|
| 4919 |
-
$objectClassExists = class_exists('phpQueryObjectPlugin_'.$class);
|
| 4920 |
-
$staticClassExists = class_exists('phpQueryPlugin_'.$class);
|
| 4921 |
-
if (! $objectClassExists && ! $staticClassExists)
|
| 4922 |
-
require_once($file);
|
| 4923 |
-
self::$pluginsLoaded[] = $class;
|
| 4924 |
-
// static methods
|
| 4925 |
-
if (class_exists('phpQueryPlugin_'.$class)) {
|
| 4926 |
-
$realClass = 'phpQueryPlugin_'.$class;
|
| 4927 |
-
$vars = get_class_vars($realClass);
|
| 4928 |
-
$loop = isset($vars['phpQueryMethods'])
|
| 4929 |
-
&& ! is_null($vars['phpQueryMethods'])
|
| 4930 |
-
? $vars['phpQueryMethods']
|
| 4931 |
-
: get_class_methods($realClass);
|
| 4932 |
-
foreach($loop as $method) {
|
| 4933 |
-
if ($method == '__initialize')
|
| 4934 |
-
continue;
|
| 4935 |
-
if (! is_callable(array($realClass, $method)))
|
| 4936 |
-
continue;
|
| 4937 |
-
if (isset(self::$pluginsStaticMethods[$method])) {
|
| 4938 |
-
throw new Exception("Duplicate method '{$method}' from plugin '{$c}' conflicts with same method from plugin '".self::$pluginsStaticMethods[$method]."'");
|
| 4939 |
-
return;
|
| 4940 |
-
}
|
| 4941 |
-
self::$pluginsStaticMethods[$method] = $class;
|
| 4942 |
-
}
|
| 4943 |
-
if (method_exists($realClass, '__initialize'))
|
| 4944 |
-
call_user_func_array(array($realClass, '__initialize'), array());
|
| 4945 |
-
}
|
| 4946 |
-
// object methods
|
| 4947 |
-
if (class_exists('phpQueryObjectPlugin_'.$class)) {
|
| 4948 |
-
$realClass = 'phpQueryObjectPlugin_'.$class;
|
| 4949 |
-
$vars = get_class_vars($realClass);
|
| 4950 |
-
$loop = isset($vars['phpQueryMethods'])
|
| 4951 |
-
&& ! is_null($vars['phpQueryMethods'])
|
| 4952 |
-
? $vars['phpQueryMethods']
|
| 4953 |
-
: get_class_methods($realClass);
|
| 4954 |
-
foreach($loop as $method) {
|
| 4955 |
-
if (! is_callable(array($realClass, $method)))
|
| 4956 |
-
continue;
|
| 4957 |
-
if (isset(self::$pluginsMethods[$method])) {
|
| 4958 |
-
throw new Exception("Duplicate method '{$method}' from plugin '{$c}' conflicts with same method from plugin '".self::$pluginsMethods[$method]."'");
|
| 4959 |
-
continue;
|
| 4960 |
-
}
|
| 4961 |
-
self::$pluginsMethods[$method] = $class;
|
| 4962 |
-
}
|
| 4963 |
-
}
|
| 4964 |
-
return true;
|
| 4965 |
-
}
|
| 4966 |
-
/**
|
| 4967 |
-
* Unloades all or specified document from memory.
|
| 4968 |
-
*
|
| 4969 |
-
* @param mixed $documentID @see phpQuery::getDocumentID() for supported types.
|
| 4970 |
-
*/
|
| 4971 |
-
public static function unloadDocuments($id = null) {
|
| 4972 |
-
if (isset($id)) {
|
| 4973 |
-
if ($id = self::getDocumentID($id))
|
| 4974 |
-
unset(phpQuery::$documents[$id]);
|
| 4975 |
-
} else {
|
| 4976 |
-
foreach(phpQuery::$documents as $k => $v) {
|
| 4977 |
-
unset(phpQuery::$documents[$k]);
|
| 4978 |
-
}
|
| 4979 |
-
}
|
| 4980 |
-
}
|
| 4981 |
-
/**
|
| 4982 |
-
* Parses phpQuery object or HTML result against PHP tags and makes them active.
|
| 4983 |
-
*
|
| 4984 |
-
* @param phpQuery|string $content
|
| 4985 |
-
* @deprecated
|
| 4986 |
-
* @return string
|
| 4987 |
-
*/
|
| 4988 |
-
public static function unsafePHPTags($content) {
|
| 4989 |
-
return self::markupToPHP($content);
|
| 4990 |
-
}
|
| 4991 |
-
public static function DOMNodeListToArray($DOMNodeList) {
|
| 4992 |
-
$array = array();
|
| 4993 |
-
if (! $DOMNodeList)
|
| 4994 |
-
return $array;
|
| 4995 |
-
foreach($DOMNodeList as $node)
|
| 4996 |
-
$array[] = $node;
|
| 4997 |
-
return $array;
|
| 4998 |
-
}
|
| 4999 |
-
/**
|
| 5000 |
-
* Checks if $input is HTML string, which has to start with '<'.
|
| 5001 |
-
*
|
| 5002 |
-
* @deprecated
|
| 5003 |
-
* @param String $input
|
| 5004 |
-
* @return Bool
|
| 5005 |
-
* @todo still used ?
|
| 5006 |
-
*/
|
| 5007 |
-
public static function isMarkup($input) {
|
| 5008 |
-
return ! is_array($input) && substr(trim($input), 0, 1) == '<';
|
| 5009 |
-
}
|
| 5010 |
-
public static function debug($text) {
|
| 5011 |
-
if (self::$debug)
|
| 5012 |
-
print var_dump($text);
|
| 5013 |
-
}
|
| 5014 |
-
/**
|
| 5015 |
-
* Make an AJAX request.
|
| 5016 |
-
*
|
| 5017 |
-
* @param array See $options http://docs.jquery.com/Ajax/jQuery.ajax#toptions
|
| 5018 |
-
* Additional options are:
|
| 5019 |
-
* 'document' - document for global events, @see phpQuery::getDocumentID()
|
| 5020 |
-
* 'referer' - implemented
|
| 5021 |
-
* 'requested_with' - TODO; not implemented (X-Requested-With)
|
| 5022 |
-
* @return Zend_Http_Client
|
| 5023 |
-
* @link http://docs.jquery.com/Ajax/jQuery.ajax
|
| 5024 |
-
*
|
| 5025 |
-
* @TODO $options['cache']
|
| 5026 |
-
* @TODO $options['processData']
|
| 5027 |
-
* @TODO $options['xhr']
|
| 5028 |
-
* @TODO $options['data'] as string
|
| 5029 |
-
* @TODO XHR interface
|
| 5030 |
-
*/
|
| 5031 |
-
public static function ajax($options = array(), $xhr = null) {
|
| 5032 |
-
$options = array_merge(
|
| 5033 |
-
self::$ajaxSettings, $options
|
| 5034 |
-
);
|
| 5035 |
-
$documentID = isset($options['document'])
|
| 5036 |
-
? self::getDocumentID($options['document'])
|
| 5037 |
-
: null;
|
| 5038 |
-
if ($xhr) {
|
| 5039 |
-
// reuse existing XHR object, but clean it up
|
| 5040 |
-
$client = $xhr;
|
| 5041 |
-
// $client->setParameterPost(null);
|
| 5042 |
-
// $client->setParameterGet(null);
|
| 5043 |
-
$client->setAuth(false);
|
| 5044 |
-
$client->setHeaders("If-Modified-Since", null);
|
| 5045 |
-
$client->setHeaders("Referer", null);
|
| 5046 |
-
$client->resetParameters();
|
| 5047 |
-
} else {
|
| 5048 |
-
// create new XHR object
|
| 5049 |
-
require_once('Zend/Http/Client.php');
|
| 5050 |
-
$client = new Zend_Http_Client();
|
| 5051 |
-
$client->setCookieJar();
|
| 5052 |
-
}
|
| 5053 |
-
if (isset($options['timeout']))
|
| 5054 |
-
$client->setConfig(array(
|
| 5055 |
-
'timeout' => $options['timeout'],
|
| 5056 |
-
));
|
| 5057 |
-
// 'maxredirects' => 0,
|
| 5058 |
-
foreach(self::$ajaxAllowedHosts as $k => $host)
|
| 5059 |
-
if ($host == '.' && isset($_SERVER['HTTP_HOST']))
|
| 5060 |
-
self::$ajaxAllowedHosts[$k] = $_SERVER['HTTP_HOST'];
|
| 5061 |
-
$host = parse_url($options['url'], PHP_URL_HOST);
|
| 5062 |
-
if (! in_array($host, self::$ajaxAllowedHosts)) {
|
| 5063 |
-
throw new Exception("Request not permitted, host '$host' not present in "
|
| 5064 |
-
."phpQuery::\$ajaxAllowedHosts");
|
| 5065 |
-
}
|
| 5066 |
-
// JSONP
|
| 5067 |
-
$jsre = "/=\\?(&|$)/";
|
| 5068 |
-
if (isset($options['dataType']) && $options['dataType'] == 'jsonp') {
|
| 5069 |
-
$jsonpCallbackParam = $options['jsonp']
|
| 5070 |
-
? $options['jsonp'] : 'callback';
|
| 5071 |
-
if (strtolower($options['type']) == 'get') {
|
| 5072 |
-
if (! preg_match($jsre, $options['url'])) {
|
| 5073 |
-
$sep = strpos($options['url'], '?')
|
| 5074 |
-
? '&' : '?';
|
| 5075 |
-
$options['url'] .= "$sep$jsonpCallbackParam=?";
|
| 5076 |
-
}
|
| 5077 |
-
} else if ($options['data']) {
|
| 5078 |
-
$jsonp = false;
|
| 5079 |
-
foreach($options['data'] as $n => $v) {
|
| 5080 |
-
if ($v == '?')
|
| 5081 |
-
$jsonp = true;
|
| 5082 |
-
}
|
| 5083 |
-
if (! $jsonp) {
|
| 5084 |
-
$options['data'][$jsonpCallbackParam] = '?';
|
| 5085 |
-
}
|
| 5086 |
-
}
|
| 5087 |
-
$options['dataType'] = 'json';
|
| 5088 |
-
}
|
| 5089 |
-
if (isset($options['dataType']) && $options['dataType'] == 'json') {
|
| 5090 |
-
$jsonpCallback = 'json_'.md5(microtime());
|
| 5091 |
-
$jsonpData = $jsonpUrl = false;
|
| 5092 |
-
if ($options['data']) {
|
| 5093 |
-
foreach($options['data'] as $n => $v) {
|
| 5094 |
-
if ($v == '?')
|
| 5095 |
-
$jsonpData = $n;
|
| 5096 |
-
}
|
| 5097 |
-
}
|
| 5098 |
-
if (preg_match($jsre, $options['url']))
|
| 5099 |
-
$jsonpUrl = true;
|
| 5100 |
-
if ($jsonpData !== false || $jsonpUrl) {
|
| 5101 |
-
// remember callback name for httpData()
|
| 5102 |
-
$options['_jsonp'] = $jsonpCallback;
|
| 5103 |
-
if ($jsonpData !== false)
|
| 5104 |
-
$options['data'][$jsonpData] = $jsonpCallback;
|
| 5105 |
-
if ($jsonpUrl)
|
| 5106 |
-
$options['url'] = preg_replace($jsre, "=$jsonpCallback\\1", $options['url']);
|
| 5107 |
-
}
|
| 5108 |
-
}
|
| 5109 |
-
$client->setUri($options['url']);
|
| 5110 |
-
$client->setMethod(strtoupper($options['type']));
|
| 5111 |
-
if (isset($options['referer']) && $options['referer'])
|
| 5112 |
-
$client->setHeaders('Referer', $options['referer']);
|
| 5113 |
-
$client->setHeaders(array(
|
| 5114 |
-
// 'content-type' => $options['contentType'],
|
| 5115 |
-
'User-Agent' => 'Mozilla/5.0 (X11; U; Linux x86; en-US; rv:1.9.0.5) Gecko'
|
| 5116 |
-
.'/2008122010 Firefox/3.0.5',
|
| 5117 |
-
// TODO custom charset
|
| 5118 |
-
'Accept-Charset' => 'ISO-8859-1,utf-8;q=0.7,*;q=0.7',
|
| 5119 |
-
// 'Connection' => 'keep-alive',
|
| 5120 |
-
// 'Accept' => 'text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8',
|
| 5121 |
-
'Accept-Language' => 'en-us,en;q=0.5',
|
| 5122 |
-
));
|
| 5123 |
-
if ($options['username'])
|
| 5124 |
-
$client->setAuth($options['username'], $options['password']);
|
| 5125 |
-
if (isset($options['ifModified']) && $options['ifModified'])
|
| 5126 |
-
$client->setHeaders("If-Modified-Since",
|
| 5127 |
-
self::$lastModified
|
| 5128 |
-
? self::$lastModified
|
| 5129 |
-
: "Thu, 01 Jan 1970 00:00:00 GMT"
|
| 5130 |
-
);
|
| 5131 |
-
$client->setHeaders("Accept",
|
| 5132 |
-
isset($options['dataType'])
|
| 5133 |
-
&& isset(self::$ajaxSettings['accepts'][ $options['dataType'] ])
|
| 5134 |
-
? self::$ajaxSettings['accepts'][ $options['dataType'] ].", */*"
|
| 5135 |
-
: self::$ajaxSettings['accepts']['_default']
|
| 5136 |
-
);
|
| 5137 |
-
// TODO $options['processData']
|
| 5138 |
-
if ($options['data'] instanceof phpQueryObject) {
|
| 5139 |
-
$serialized = $options['data']->serializeArray($options['data']);
|
| 5140 |
-
$options['data'] = array();
|
| 5141 |
-
foreach($serialized as $r)
|
| 5142 |
-
$options['data'][ $r['name'] ] = $r['value'];
|
| 5143 |
-
}
|
| 5144 |
-
if (strtolower($options['type']) == 'get') {
|
| 5145 |
-
$client->setParameterGet($options['data']);
|
| 5146 |
-
} else if (strtolower($options['type']) == 'post') {
|
| 5147 |
-
$client->setEncType($options['contentType']);
|
| 5148 |
-
$client->setParameterPost($options['data']);
|
| 5149 |
-
}
|
| 5150 |
-
if (self::$active == 0 && $options['global'])
|
| 5151 |
-
phpQueryEvents::trigger($documentID, 'ajaxStart');
|
| 5152 |
-
self::$active++;
|
| 5153 |
-
// beforeSend callback
|
| 5154 |
-
if (isset($options['beforeSend']) && $options['beforeSend'])
|
| 5155 |
-
phpQuery::callbackRun($options['beforeSend'], array($client));
|
| 5156 |
-
// ajaxSend event
|
| 5157 |
-
if ($options['global'])
|
| 5158 |
-
phpQueryEvents::trigger($documentID, 'ajaxSend', array($client, $options));
|
| 5159 |
-
if (phpQuery::$debug) {
|
| 5160 |
-
self::debug("{$options['type']}: {$options['url']}\n");
|
| 5161 |
-
self::debug("Options: <pre>".var_export($options, true)."</pre>\n");
|
| 5162 |
-
// if ($client->getCookieJar())
|
| 5163 |
-
// self::debug("Cookies: <pre>".var_export($client->getCookieJar()->getMatchingCookies($options['url']), true)."</pre>\n");
|
| 5164 |
-
}
|
| 5165 |
-
// request
|
| 5166 |
-
$response = $client->request();
|
| 5167 |
-
if (phpQuery::$debug) {
|
| 5168 |
-
self::debug('Status: '.$response->getStatus().' / '.$response->getMessage());
|
| 5169 |
-
self::debug($client->getLastRequest());
|
| 5170 |
-
self::debug($response->getHeaders());
|
| 5171 |
-
}
|
| 5172 |
-
if ($response->isSuccessful()) {
|
| 5173 |
-
// XXX tempolary
|
| 5174 |
-
self::$lastModified = $response->getHeader('Last-Modified');
|
| 5175 |
-
$data = self::httpData($response->getBody(), $options['dataType'], $options);
|
| 5176 |
-
if (isset($options['success']) && $options['success'])
|
| 5177 |
-
phpQuery::callbackRun($options['success'], array($data, $response->getStatus(), $options));
|
| 5178 |
-
if ($options['global'])
|
| 5179 |
-
phpQueryEvents::trigger($documentID, 'ajaxSuccess', array($client, $options));
|
| 5180 |
-
} else {
|
| 5181 |
-
if (isset($options['error']) && $options['error'])
|
| 5182 |
-
phpQuery::callbackRun($options['error'], array($client, $response->getStatus(), $response->getMessage()));
|
| 5183 |
-
if ($options['global'])
|
| 5184 |
-
phpQueryEvents::trigger($documentID, 'ajaxError', array($client, /*$response->getStatus(),*/$response->getMessage(), $options));
|
| 5185 |
-
}
|
| 5186 |
-
if (isset($options['complete']) && $options['complete'])
|
| 5187 |
-
phpQuery::callbackRun($options['complete'], array($client, $response->getStatus()));
|
| 5188 |
-
if ($options['global'])
|
| 5189 |
-
phpQueryEvents::trigger($documentID, 'ajaxComplete', array($client, $options));
|
| 5190 |
-
if ($options['global'] && ! --self::$active)
|
| 5191 |
-
phpQueryEvents::trigger($documentID, 'ajaxStop');
|
| 5192 |
-
return $client;
|
| 5193 |
-
// if (is_null($domId))
|
| 5194 |
-
// $domId = self::$defaultDocumentID ? self::$defaultDocumentID : false;
|
| 5195 |
-
// return new phpQueryAjaxResponse($response, $domId);
|
| 5196 |
-
}
|
| 5197 |
-
protected static function httpData($data, $type, $options) {
|
| 5198 |
-
if (isset($options['dataFilter']) && $options['dataFilter'])
|
| 5199 |
-
$data = self::callbackRun($options['dataFilter'], array($data, $type));
|
| 5200 |
-
if (is_string($data)) {
|
| 5201 |
-
if ($type == "json") {
|
| 5202 |
-
if (isset($options['_jsonp']) && $options['_jsonp']) {
|
| 5203 |
-
$data = preg_replace('/^\s*\w+\((.*)\)\s*$/s', '$1', $data);
|
| 5204 |
-
}
|
| 5205 |
-
$data = self::parseJSON($data);
|
| 5206 |
-
}
|
| 5207 |
-
}
|
| 5208 |
-
return $data;
|
| 5209 |
-
}
|
| 5210 |
-
/**
|
| 5211 |
-
* Enter description here...
|
| 5212 |
-
*
|
| 5213 |
-
* @param array|phpQuery $data
|
| 5214 |
-
*
|
| 5215 |
-
*/
|
| 5216 |
-
public static function param($data) {
|
| 5217 |
-
return http_build_query($data, null, '&');
|
| 5218 |
-
}
|
| 5219 |
-
public static function get($url, $data = null, $callback = null, $type = null) {
|
| 5220 |
-
if (!is_array($data)) {
|
| 5221 |
-
$callback = $data;
|
| 5222 |
-
$data = null;
|
| 5223 |
-
}
|
| 5224 |
-
// TODO some array_values on this shit
|
| 5225 |
-
return phpQuery::ajax(array(
|
| 5226 |
-
'type' => 'GET',
|
| 5227 |
-
'url' => $url,
|
| 5228 |
-
'data' => $data,
|
| 5229 |
-
'success' => $callback,
|
| 5230 |
-
'dataType' => $type,
|
| 5231 |
-
));
|
| 5232 |
-
}
|
| 5233 |
-
public static function post($url, $data = null, $callback = null, $type = null) {
|
| 5234 |
-
if (!is_array($data)) {
|
| 5235 |
-
$callback = $data;
|
| 5236 |
-
$data = null;
|
| 5237 |
-
}
|
| 5238 |
-
return phpQuery::ajax(array(
|
| 5239 |
-
'type' => 'POST',
|
| 5240 |
-
'url' => $url,
|
| 5241 |
-
'data' => $data,
|
| 5242 |
-
'success' => $callback,
|
| 5243 |
-
'dataType' => $type,
|
| 5244 |
-
));
|
| 5245 |
-
}
|
| 5246 |
-
public static function getJSON($url, $data = null, $callback = null) {
|
| 5247 |
-
if (!is_array($data)) {
|
| 5248 |
-
$callback = $data;
|
| 5249 |
-
$data = null;
|
| 5250 |
-
}
|
| 5251 |
-
// TODO some array_values on this shit
|
| 5252 |
-
return phpQuery::ajax(array(
|
| 5253 |
-
'type' => 'GET',
|
| 5254 |
-
'url' => $url,
|
| 5255 |
-
'data' => $data,
|
| 5256 |
-
'success' => $callback,
|
| 5257 |
-
'dataType' => 'json',
|
| 5258 |
-
));
|
| 5259 |
-
}
|
| 5260 |
-
public static function ajaxSetup($options) {
|
| 5261 |
-
self::$ajaxSettings = array_merge(
|
| 5262 |
-
self::$ajaxSettings,
|
| 5263 |
-
$options
|
| 5264 |
-
);
|
| 5265 |
-
}
|
| 5266 |
-
public static function ajaxAllowHost($host1, $host2 = null, $host3 = null) {
|
| 5267 |
-
$loop = is_array($host1)
|
| 5268 |
-
? $host1
|
| 5269 |
-
: func_get_args();
|
| 5270 |
-
foreach($loop as $host) {
|
| 5271 |
-
if ($host && ! in_array($host, phpQuery::$ajaxAllowedHosts)) {
|
| 5272 |
-
phpQuery::$ajaxAllowedHosts[] = $host;
|
| 5273 |
-
}
|
| 5274 |
-
}
|
| 5275 |
-
}
|
| 5276 |
-
public static function ajaxAllowURL($url1, $url2 = null, $url3 = null) {
|
| 5277 |
-
$loop = is_array($url1)
|
| 5278 |
-
? $url1
|
| 5279 |
-
: func_get_args();
|
| 5280 |
-
foreach($loop as $url)
|
| 5281 |
-
phpQuery::ajaxAllowHost(parse_url($url, PHP_URL_HOST));
|
| 5282 |
-
}
|
| 5283 |
-
/**
|
| 5284 |
-
* Returns JSON representation of $data.
|
| 5285 |
-
*
|
| 5286 |
-
* @static
|
| 5287 |
-
* @param mixed $data
|
| 5288 |
-
* @return string
|
| 5289 |
-
*/
|
| 5290 |
-
public static function toJSON($data) {
|
| 5291 |
-
if (function_exists('json_encode'))
|
| 5292 |
-
return json_encode($data);
|
| 5293 |
-
require_once('Zend/Json/Encoder.php');
|
| 5294 |
-
return Zend_Json_Encoder::encode($data);
|
| 5295 |
-
}
|
| 5296 |
-
/**
|
| 5297 |
-
* Parses JSON into proper PHP type.
|
| 5298 |
-
*
|
| 5299 |
-
* @static
|
| 5300 |
-
* @param string $json
|
| 5301 |
-
* @return mixed
|
| 5302 |
-
*/
|
| 5303 |
-
public static function parseJSON($json) {
|
| 5304 |
-
if (function_exists('json_decode')) {
|
| 5305 |
-
$return = json_decode(trim($json), true);
|
| 5306 |
-
// json_decode and UTF8 issues
|
| 5307 |
-
if (isset($return))
|
| 5308 |
-
return $return;
|
| 5309 |
-
}
|
| 5310 |
-
require_once('Zend/Json/Decoder.php');
|
| 5311 |
-
return Zend_Json_Decoder::decode($json);
|
| 5312 |
-
}
|
| 5313 |
-
/**
|
| 5314 |
-
* Returns source's document ID.
|
| 5315 |
-
*
|
| 5316 |
-
* @param $source DOMNode|phpQueryObject
|
| 5317 |
-
* @return string
|
| 5318 |
-
*/
|
| 5319 |
-
public static function getDocumentID($source) {
|
| 5320 |
-
if ($source instanceof DOMDOCUMENT) {
|
| 5321 |
-
foreach(phpQuery::$documents as $id => $document) {
|
| 5322 |
-
if ($source->isSameNode($document->document))
|
| 5323 |
-
return $id;
|
| 5324 |
-
}
|
| 5325 |
-
} else if ($source instanceof DOMNODE) {
|
| 5326 |
-
foreach(phpQuery::$documents as $id => $document) {
|
| 5327 |
-
if ($source->ownerDocument->isSameNode($document->document))
|
| 5328 |
-
return $id;
|
| 5329 |
-
}
|
| 5330 |
-
} else if ($source instanceof phpQueryObject)
|
| 5331 |
-
return $source->getDocumentID();
|
| 5332 |
-
else if (is_string($source) && isset(phpQuery::$documents[$source]))
|
| 5333 |
-
return $source;
|
| 5334 |
-
}
|
| 5335 |
-
/**
|
| 5336 |
-
* Get DOMDocument object related to $source.
|
| 5337 |
-
* Returns null if such document doesn't exist.
|
| 5338 |
-
*
|
| 5339 |
-
* @param $source DOMNode|phpQueryObject|string
|
| 5340 |
-
* @return string
|
| 5341 |
-
*/
|
| 5342 |
-
public static function getDOMDocument($source) {
|
| 5343 |
-
if ($source instanceof DOMDOCUMENT)
|
| 5344 |
-
return $source;
|
| 5345 |
-
$source = self::getDocumentID($source);
|
| 5346 |
-
return $source
|
| 5347 |
-
? self::$documents[$id]['document']
|
| 5348 |
-
: null;
|
| 5349 |
-
}
|
| 5350 |
-
|
| 5351 |
-
// UTILITIES
|
| 5352 |
-
// http://docs.jquery.com/Utilities
|
| 5353 |
-
|
| 5354 |
-
/**
|
| 5355 |
-
*
|
| 5356 |
-
* @return unknown_type
|
| 5357 |
-
* @link http://docs.jquery.com/Utilities/jQuery.makeArray
|
| 5358 |
-
*/
|
| 5359 |
-
public static function makeArray($obj) {
|
| 5360 |
-
$array = array();
|
| 5361 |
-
if (is_object($object) && $object instanceof DOMNODELIST) {
|
| 5362 |
-
foreach($object as $value)
|
| 5363 |
-
$array[] = $value;
|
| 5364 |
-
} else if (is_object($object) && ! ($object instanceof Iterator)) {
|
| 5365 |
-
foreach(get_object_vars($object) as $name => $value)
|
| 5366 |
-
$array[0][$name] = $value;
|
| 5367 |
-
} else {
|
| 5368 |
-
foreach($object as $name => $value)
|
| 5369 |
-
$array[0][$name] = $value;
|
| 5370 |
-
}
|
| 5371 |
-
return $array;
|
| 5372 |
-
}
|
| 5373 |
-
public static function inArray($value, $array) {
|
| 5374 |
-
return in_array($value, $array);
|
| 5375 |
-
}
|
| 5376 |
-
/**
|
| 5377 |
-
*
|
| 5378 |
-
* @param $object
|
| 5379 |
-
* @param $callback
|
| 5380 |
-
* @return unknown_type
|
| 5381 |
-
* @link http://docs.jquery.com/Utilities/jQuery.each
|
| 5382 |
-
*/
|
| 5383 |
-
public static function each($object, $callback, $param1 = null, $param2 = null, $param3 = null) {
|
| 5384 |
-
$paramStructure = null;
|
| 5385 |
-
if (func_num_args() > 2) {
|
| 5386 |
-
$paramStructure = func_get_args();
|
| 5387 |
-
$paramStructure = array_slice($paramStructure, 2);
|
| 5388 |
-
}
|
| 5389 |
-
if (is_object($object) && ! ($object instanceof Iterator)) {
|
| 5390 |
-
foreach(get_object_vars($object) as $name => $value)
|
| 5391 |
-
phpQuery::callbackRun($callback, array($name, $value), $paramStructure);
|
| 5392 |
-
} else {
|
| 5393 |
-
foreach($object as $name => $value)
|
| 5394 |
-
phpQuery::callbackRun($callback, array($name, $value), $paramStructure);
|
| 5395 |
-
}
|
| 5396 |
-
}
|
| 5397 |
-
/**
|
| 5398 |
-
*
|
| 5399 |
-
* @link http://docs.jquery.com/Utilities/jQuery.map
|
| 5400 |
-
*/
|
| 5401 |
-
public static function map($array, $callback, $param1 = null, $param2 = null, $param3 = null) {
|
| 5402 |
-
$result = array();
|
| 5403 |
-
$paramStructure = null;
|
| 5404 |
-
if (func_num_args() > 2) {
|
| 5405 |
-
$paramStructure = func_get_args();
|
| 5406 |
-
$paramStructure = array_slice($paramStructure, 2);
|
| 5407 |
-
}
|
| 5408 |
-
foreach($array as $v) {
|
| 5409 |
-
$vv = phpQuery::callbackRun($callback, array($v), $paramStructure);
|
| 5410 |
-
// $callbackArgs = $args;
|
| 5411 |
-
// foreach($args as $i => $arg) {
|
| 5412 |
-
// $callbackArgs[$i] = $arg instanceof CallbackParam
|
| 5413 |
-
// ? $v
|
| 5414 |
-
// : $arg;
|
| 5415 |
-
// }
|
| 5416 |
-
// $vv = call_user_func_array($callback, $callbackArgs);
|
| 5417 |
-
if (is_array($vv)) {
|
| 5418 |
-
foreach($vv as $vvv)
|
| 5419 |
-
$result[] = $vvv;
|
| 5420 |
-
} else if ($vv !== null) {
|
| 5421 |
-
$result[] = $vv;
|
| 5422 |
-
}
|
| 5423 |
-
}
|
| 5424 |
-
return $result;
|
| 5425 |
-
}
|
| 5426 |
-
/**
|
| 5427 |
-
*
|
| 5428 |
-
* @param $callback Callback
|
| 5429 |
-
* @param $params
|
| 5430 |
-
* @param $paramStructure
|
| 5431 |
-
* @return unknown_type
|
| 5432 |
-
*/
|
| 5433 |
-
public static function callbackRun($callback, $params = array(), $paramStructure = null) {
|
| 5434 |
-
if (! $callback)
|
| 5435 |
-
return;
|
| 5436 |
-
if ($callback instanceof CallbackParameterToReference) {
|
| 5437 |
-
// TODO support ParamStructure to select which $param push to reference
|
| 5438 |
-
if (isset($params[0]))
|
| 5439 |
-
$callback->callback = $params[0];
|
| 5440 |
-
return true;
|
| 5441 |
-
}
|
| 5442 |
-
if ($callback instanceof Callback) {
|
| 5443 |
-
$paramStructure = $callback->params;
|
| 5444 |
-
$callback = $callback->callback;
|
| 5445 |
-
}
|
| 5446 |
-
if (! $paramStructure)
|
| 5447 |
-
return call_user_func_array($callback, $params);
|
| 5448 |
-
$p = 0;
|
| 5449 |
-
foreach($paramStructure as $i => $v) {
|
| 5450 |
-
$paramStructure[$i] = $v instanceof CallbackParam
|
| 5451 |
-
? $params[$p++]
|
| 5452 |
-
: $v;
|
| 5453 |
-
}
|
| 5454 |
-
return call_user_func_array($callback, $paramStructure);
|
| 5455 |
-
}
|
| 5456 |
-
/**
|
| 5457 |
-
* Merge 2 phpQuery objects.
|
| 5458 |
-
* @param array $one
|
| 5459 |
-
* @param array $two
|
| 5460 |
-
* @protected
|
| 5461 |
-
* @todo node lists, phpQueryObject
|
| 5462 |
-
*/
|
| 5463 |
-
public static function merge($one, $two) {
|
| 5464 |
-
$elements = $one->elements;
|
| 5465 |
-
foreach($two->elements as $node) {
|
| 5466 |
-
$exists = false;
|
| 5467 |
-
foreach($elements as $node2) {
|
| 5468 |
-
if ($node2->isSameNode($node))
|
| 5469 |
-
$exists = true;
|
| 5470 |
-
}
|
| 5471 |
-
if (! $exists)
|
| 5472 |
-
$elements[] = $node;
|
| 5473 |
-
}
|
| 5474 |
-
return $elements;
|
| 5475 |
-
// $one = $one->newInstance();
|
| 5476 |
-
// $one->elements = $elements;
|
| 5477 |
-
// return $one;
|
| 5478 |
-
}
|
| 5479 |
-
/**
|
| 5480 |
-
*
|
| 5481 |
-
* @param $array
|
| 5482 |
-
* @param $callback
|
| 5483 |
-
* @param $invert
|
| 5484 |
-
* @return unknown_type
|
| 5485 |
-
* @link http://docs.jquery.com/Utilities/jQuery.grep
|
| 5486 |
-
*/
|
| 5487 |
-
public static function grep($array, $callback, $invert = false) {
|
| 5488 |
-
$result = array();
|
| 5489 |
-
foreach($array as $k => $v) {
|
| 5490 |
-
$r = call_user_func_array($callback, array($v, $k));
|
| 5491 |
-
if ($r === !(bool)$invert)
|
| 5492 |
-
$result[] = $v;
|
| 5493 |
-
}
|
| 5494 |
-
return $result;
|
| 5495 |
-
}
|
| 5496 |
-
public static function unique($array) {
|
| 5497 |
-
return array_unique($array);
|
| 5498 |
-
}
|
| 5499 |
-
/**
|
| 5500 |
-
*
|
| 5501 |
-
* @param $function
|
| 5502 |
-
* @return unknown_type
|
| 5503 |
-
* @TODO there are problems with non-static methods, second parameter pass it
|
| 5504 |
-
* but doesnt verify is method is really callable
|
| 5505 |
-
*/
|
| 5506 |
-
public static function isFunction($function) {
|
| 5507 |
-
return is_callable($function);
|
| 5508 |
-
}
|
| 5509 |
-
public static function trim($str) {
|
| 5510 |
-
return trim($str);
|
| 5511 |
-
}
|
| 5512 |
-
/* PLUGINS NAMESPACE */
|
| 5513 |
-
/**
|
| 5514 |
-
*
|
| 5515 |
-
* @param $url
|
| 5516 |
-
* @param $callback
|
| 5517 |
-
* @param $param1
|
| 5518 |
-
* @param $param2
|
| 5519 |
-
* @param $param3
|
| 5520 |
-
* @return phpQueryObject
|
| 5521 |
-
*/
|
| 5522 |
-
public static function browserGet($url, $callback, $param1 = null, $param2 = null, $param3 = null) {
|
| 5523 |
-
if (self::plugin('WebBrowser')) {
|
| 5524 |
-
$params = func_get_args();
|
| 5525 |
-
return self::callbackRun(array(self::$plugins, 'browserGet'), $params);
|
| 5526 |
-
} else {
|
| 5527 |
-
self::debug('WebBrowser plugin not available...');
|
| 5528 |
-
}
|
| 5529 |
-
}
|
| 5530 |
-
/**
|
| 5531 |
-
*
|
| 5532 |
-
* @param $url
|
| 5533 |
-
* @param $data
|
| 5534 |
-
* @param $callback
|
| 5535 |
-
* @param $param1
|
| 5536 |
-
* @param $param2
|
| 5537 |
-
* @param $param3
|
| 5538 |
-
* @return phpQueryObject
|
| 5539 |
-
*/
|
| 5540 |
-
public static function browserPost($url, $data, $callback, $param1 = null, $param2 = null, $param3 = null) {
|
| 5541 |
-
if (self::plugin('WebBrowser')) {
|
| 5542 |
-
$params = func_get_args();
|
| 5543 |
-
return self::callbackRun(array(self::$plugins, 'browserPost'), $params);
|
| 5544 |
-
} else {
|
| 5545 |
-
self::debug('WebBrowser plugin not available...');
|
| 5546 |
-
}
|
| 5547 |
-
}
|
| 5548 |
-
/**
|
| 5549 |
-
*
|
| 5550 |
-
* @param $ajaxSettings
|
| 5551 |
-
* @param $callback
|
| 5552 |
-
* @param $param1
|
| 5553 |
-
* @param $param2
|
| 5554 |
-
* @param $param3
|
| 5555 |
-
* @return phpQueryObject
|
| 5556 |
-
*/
|
| 5557 |
-
public static function browser($ajaxSettings, $callback, $param1 = null, $param2 = null, $param3 = null) {
|
| 5558 |
-
if (self::plugin('WebBrowser')) {
|
| 5559 |
-
$params = func_get_args();
|
| 5560 |
-
return self::callbackRun(array(self::$plugins, 'browser'), $params);
|
| 5561 |
-
} else {
|
| 5562 |
-
self::debug('WebBrowser plugin not available...');
|
| 5563 |
-
}
|
| 5564 |
-
}
|
| 5565 |
-
/**
|
| 5566 |
-
*
|
| 5567 |
-
* @param $code
|
| 5568 |
-
* @return string
|
| 5569 |
-
*/
|
| 5570 |
-
public static function php($code) {
|
| 5571 |
-
return self::code('php', $code);
|
| 5572 |
-
}
|
| 5573 |
-
/**
|
| 5574 |
-
*
|
| 5575 |
-
* @param $type
|
| 5576 |
-
* @param $code
|
| 5577 |
-
* @return string
|
| 5578 |
-
*/
|
| 5579 |
-
public static function code($type, $code) {
|
| 5580 |
-
return "<$type><!-- ".trim($code)." --></$type>";
|
| 5581 |
-
}
|
| 5582 |
-
|
| 5583 |
-
public static function __callStatic($method, $params) {
|
| 5584 |
-
return call_user_func_array(
|
| 5585 |
-
array(phpQuery::$plugins, $method),
|
| 5586 |
-
$params
|
| 5587 |
-
);
|
| 5588 |
-
}
|
| 5589 |
-
protected static function dataSetupNode($node, $documentID) {
|
| 5590 |
-
// search are return if alredy exists
|
| 5591 |
-
foreach(phpQuery::$documents[$documentID]->dataNodes as $dataNode) {
|
| 5592 |
-
if ($node->isSameNode($dataNode))
|
| 5593 |
-
return $dataNode;
|
| 5594 |
-
}
|
| 5595 |
-
// if doesn't, add it
|
| 5596 |
-
phpQuery::$documents[$documentID]->dataNodes[] = $node;
|
| 5597 |
-
return $node;
|
| 5598 |
-
}
|
| 5599 |
-
protected static function dataRemoveNode($node, $documentID) {
|
| 5600 |
-
// search are return if alredy exists
|
| 5601 |
-
foreach(phpQuery::$documents[$documentID]->dataNodes as $k => $dataNode) {
|
| 5602 |
-
if ($node->isSameNode($dataNode)) {
|
| 5603 |
-
unset(self::$documents[$documentID]->dataNodes[$k]);
|
| 5604 |
-
unset(self::$documents[$documentID]->data[ $dataNode->dataID ]);
|
| 5605 |
-
}
|
| 5606 |
-
}
|
| 5607 |
-
}
|
| 5608 |
-
public static function data($node, $name, $data, $documentID = null) {
|
| 5609 |
-
if (! $documentID)
|
| 5610 |
-
// TODO check if this works
|
| 5611 |
-
$documentID = self::getDocumentID($node);
|
| 5612 |
-
$document = phpQuery::$documents[$documentID];
|
| 5613 |
-
$node = self::dataSetupNode($node, $documentID);
|
| 5614 |
-
if (! isset($node->dataID))
|
| 5615 |
-
$node->dataID = ++phpQuery::$documents[$documentID]->uuid;
|
| 5616 |
-
$id = $node->dataID;
|
| 5617 |
-
if (! isset($document->data[$id]))
|
| 5618 |
-
$document->data[$id] = array();
|
| 5619 |
-
if (! is_null($data))
|
| 5620 |
-
$document->data[$id][$name] = $data;
|
| 5621 |
-
if ($name) {
|
| 5622 |
-
if (isset($document->data[$id][$name]))
|
| 5623 |
-
return $document->data[$id][$name];
|
| 5624 |
-
} else
|
| 5625 |
-
return $id;
|
| 5626 |
-
}
|
| 5627 |
-
public static function removeData($node, $name, $documentID) {
|
| 5628 |
-
if (! $documentID)
|
| 5629 |
-
// TODO check if this works
|
| 5630 |
-
$documentID = self::getDocumentID($node);
|
| 5631 |
-
$document = phpQuery::$documents[$documentID];
|
| 5632 |
-
$node = self::dataSetupNode($node, $documentID);
|
| 5633 |
-
$id = $node->dataID;
|
| 5634 |
-
if ($name) {
|
| 5635 |
-
if (isset($document->data[$id][$name]))
|
| 5636 |
-
unset($document->data[$id][$name]);
|
| 5637 |
-
$name = null;
|
| 5638 |
-
foreach($document->data[$id] as $name)
|
| 5639 |
-
break;
|
| 5640 |
-
if (! $name)
|
| 5641 |
-
self::removeData($node, $name, $documentID);
|
| 5642 |
-
} else {
|
| 5643 |
-
self::dataRemoveNode($node, $documentID);
|
| 5644 |
-
}
|
| 5645 |
-
}
|
| 5646 |
-
}
|
| 5647 |
-
/**
|
| 5648 |
-
* Plugins static namespace class.
|
| 5649 |
-
*
|
| 5650 |
-
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 5651 |
-
* @package phpQuery
|
| 5652 |
-
* @todo move plugin methods here (as statics)
|
| 5653 |
-
*/
|
| 5654 |
-
class phpQueryPlugins {
|
| 5655 |
-
public function __call($method, $args) {
|
| 5656 |
-
if (isset(phpQuery::$extendStaticMethods[$method])) {
|
| 5657 |
-
$return = call_user_func_array(
|
| 5658 |
-
phpQuery::$extendStaticMethods[$method],
|
| 5659 |
-
$args
|
| 5660 |
-
);
|
| 5661 |
-
} else if (isset(phpQuery::$pluginsStaticMethods[$method])) {
|
| 5662 |
-
$class = phpQuery::$pluginsStaticMethods[$method];
|
| 5663 |
-
$realClass = "phpQueryPlugin_$class";
|
| 5664 |
-
$return = call_user_func_array(
|
| 5665 |
-
array($realClass, $method),
|
| 5666 |
-
$args
|
| 5667 |
-
);
|
| 5668 |
-
return isset($return)
|
| 5669 |
-
? $return
|
| 5670 |
-
: $this;
|
| 5671 |
-
} else
|
| 5672 |
-
throw new Exception("Method '{$method}' doesnt exist");
|
| 5673 |
-
}
|
| 5674 |
-
}
|
| 5675 |
-
/**
|
| 5676 |
-
* Shortcut to phpQuery::pq($arg1, $context)
|
| 5677 |
-
* Chainable.
|
| 5678 |
-
*
|
| 5679 |
-
* @see phpQuery::pq()
|
| 5680 |
-
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 5681 |
-
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 5682 |
-
* @package phpQuery
|
| 5683 |
-
*/
|
| 5684 |
-
function pq($arg1, $context = null) {
|
| 5685 |
-
$args = func_get_args();
|
| 5686 |
-
return call_user_func_array(
|
| 5687 |
-
array('phpQuery', 'pq'),
|
| 5688 |
-
$args
|
| 5689 |
-
);
|
| 5690 |
-
}
|
| 5691 |
-
// add plugins dir and Zend framework to include path
|
| 5692 |
-
set_include_path(
|
| 5693 |
-
get_include_path()
|
| 5694 |
-
.PATH_SEPARATOR.dirname(__FILE__).'/phpQuery/'
|
| 5695 |
-
.PATH_SEPARATOR.dirname(__FILE__).'/phpQuery/plugins/'
|
| 5696 |
-
);
|
| 5697 |
-
// why ? no __call nor __get for statics in php...
|
| 5698 |
-
// XXX __callStatic will be available in PHP 5.3
|
| 5699 |
-
phpQuery::$plugins = new phpQueryPlugins();
|
| 5700 |
-
// include bootstrap file (personal library config)
|
| 5701 |
-
if (file_exists(dirname(__FILE__).'/phpQuery/bootstrap.php'))
|
| 5702 |
require_once dirname(__FILE__).'/phpQuery/bootstrap.php';
|
| 1 |
+
<?php
|
| 2 |
+
/**
|
| 3 |
+
* phpQuery is a server-side, chainable, CSS3 selector driven
|
| 4 |
+
* Document Object Model (DOM) API based on jQuery JavaScript Library.
|
| 5 |
+
*
|
| 6 |
+
* @version 0.9.5
|
| 7 |
+
* @link http://code.google.com/p/phpquery/
|
| 8 |
+
* @link http://phpquery-library.blogspot.com/
|
| 9 |
+
* @link http://jquery.com/
|
| 10 |
+
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 11 |
+
* @license http://www.opensource.org/licenses/mit-license.php MIT License
|
| 12 |
+
* @package phpQuery
|
| 13 |
+
*/
|
| 14 |
+
|
| 15 |
+
// class names for instanceof
|
| 16 |
+
// TODO move them as class constants into phpQuery
|
| 17 |
+
define('DOMDOCUMENT', 'DOMDocument');
|
| 18 |
+
define('DOMELEMENT', 'DOMElement');
|
| 19 |
+
define('DOMNODELIST', 'DOMNodeList');
|
| 20 |
+
define('DOMNODE', 'DOMNode');
|
| 21 |
+
|
| 22 |
+
/**
|
| 23 |
+
* DOMEvent class.
|
| 24 |
+
*
|
| 25 |
+
* Based on
|
| 26 |
+
* @link http://developer.mozilla.org/En/DOM:event
|
| 27 |
+
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 28 |
+
* @package phpQuery
|
| 29 |
+
* @todo implement ArrayAccess ?
|
| 30 |
+
*/
|
| 31 |
+
class DOMEvent {
|
| 32 |
+
/**
|
| 33 |
+
* Returns a boolean indicating whether the event bubbles up through the DOM or not.
|
| 34 |
+
*
|
| 35 |
+
* @var unknown_type
|
| 36 |
+
*/
|
| 37 |
+
public $bubbles = true;
|
| 38 |
+
/**
|
| 39 |
+
* Returns a boolean indicating whether the event is cancelable.
|
| 40 |
+
*
|
| 41 |
+
* @var unknown_type
|
| 42 |
+
*/
|
| 43 |
+
public $cancelable = true;
|
| 44 |
+
/**
|
| 45 |
+
* Returns a reference to the currently registered target for the event.
|
| 46 |
+
*
|
| 47 |
+
* @var unknown_type
|
| 48 |
+
*/
|
| 49 |
+
public $currentTarget;
|
| 50 |
+
/**
|
| 51 |
+
* Returns detail about the event, depending on the type of event.
|
| 52 |
+
*
|
| 53 |
+
* @var unknown_type
|
| 54 |
+
* @link http://developer.mozilla.org/en/DOM/event.detail
|
| 55 |
+
*/
|
| 56 |
+
public $detail; // ???
|
| 57 |
+
/**
|
| 58 |
+
* Used to indicate which phase of the event flow is currently being evaluated.
|
| 59 |
+
*
|
| 60 |
+
* NOT IMPLEMENTED
|
| 61 |
+
*
|
| 62 |
+
* @var unknown_type
|
| 63 |
+
* @link http://developer.mozilla.org/en/DOM/event.eventPhase
|
| 64 |
+
*/
|
| 65 |
+
public $eventPhase; // ???
|
| 66 |
+
/**
|
| 67 |
+
* The explicit original target of the event (Mozilla-specific).
|
| 68 |
+
*
|
| 69 |
+
* NOT IMPLEMENTED
|
| 70 |
+
*
|
| 71 |
+
* @var unknown_type
|
| 72 |
+
*/
|
| 73 |
+
public $explicitOriginalTarget; // moz only
|
| 74 |
+
/**
|
| 75 |
+
* The original target of the event, before any retargetings (Mozilla-specific).
|
| 76 |
+
*
|
| 77 |
+
* NOT IMPLEMENTED
|
| 78 |
+
*
|
| 79 |
+
* @var unknown_type
|
| 80 |
+
*/
|
| 81 |
+
public $originalTarget; // moz only
|
| 82 |
+
/**
|
| 83 |
+
* Identifies a secondary target for the event.
|
| 84 |
+
*
|
| 85 |
+
* @var unknown_type
|
| 86 |
+
*/
|
| 87 |
+
public $relatedTarget;
|
| 88 |
+
/**
|
| 89 |
+
* Returns a reference to the target to which the event was originally dispatched.
|
| 90 |
+
*
|
| 91 |
+
* @var unknown_type
|
| 92 |
+
*/
|
| 93 |
+
public $target;
|
| 94 |
+
/**
|
| 95 |
+
* Returns the time that the event was created.
|
| 96 |
+
*
|
| 97 |
+
* @var unknown_type
|
| 98 |
+
*/
|
| 99 |
+
public $timeStamp;
|
| 100 |
+
/**
|
| 101 |
+
* Returns the name of the event (case-insensitive).
|
| 102 |
+
*/
|
| 103 |
+
public $type;
|
| 104 |
+
public $runDefault = true;
|
| 105 |
+
public $data = null;
|
| 106 |
+
public function __construct($data) {
|
| 107 |
+
foreach($data as $k => $v) {
|
| 108 |
+
$this->$k = $v;
|
| 109 |
+
}
|
| 110 |
+
if (! $this->timeStamp)
|
| 111 |
+
$this->timeStamp = time();
|
| 112 |
+
}
|
| 113 |
+
/**
|
| 114 |
+
* Cancels the event (if it is cancelable).
|
| 115 |
+
*
|
| 116 |
+
*/
|
| 117 |
+
public function preventDefault() {
|
| 118 |
+
$this->runDefault = false;
|
| 119 |
+
}
|
| 120 |
+
/**
|
| 121 |
+
* Stops the propagation of events further along in the DOM.
|
| 122 |
+
*
|
| 123 |
+
*/
|
| 124 |
+
public function stopPropagation() {
|
| 125 |
+
$this->bubbles = false;
|
| 126 |
+
}
|
| 127 |
+
}
|
| 128 |
+
|
| 129 |
+
|
| 130 |
+
/**
|
| 131 |
+
* DOMDocumentWrapper class simplifies work with DOMDocument.
|
| 132 |
+
*
|
| 133 |
+
* Know bug:
|
| 134 |
+
* - in XHTML fragments, <br /> changes to <br clear="none" />
|
| 135 |
+
*
|
| 136 |
+
* @todo check XML catalogs compatibility
|
| 137 |
+
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 138 |
+
* @package phpQuery
|
| 139 |
+
*/
|
| 140 |
+
class DOMDocumentWrapper {
|
| 141 |
+
/**
|
| 142 |
+
* @var DOMDocument
|
| 143 |
+
*/
|
| 144 |
+
public $document;
|
| 145 |
+
public $id;
|
| 146 |
+
/**
|
| 147 |
+
* @todo Rewrite as method and quess if null.
|
| 148 |
+
* @var unknown_type
|
| 149 |
+
*/
|
| 150 |
+
public $contentType = '';
|
| 151 |
+
public $xpath;
|
| 152 |
+
public $uuid = 0;
|
| 153 |
+
public $data = array();
|
| 154 |
+
public $dataNodes = array();
|
| 155 |
+
public $events = array();
|
| 156 |
+
public $eventsNodes = array();
|
| 157 |
+
public $eventsGlobal = array();
|
| 158 |
+
/**
|
| 159 |
+
* @TODO iframes support http://code.google.com/p/phpquery/issues/detail?id=28
|
| 160 |
+
* @var unknown_type
|
| 161 |
+
*/
|
| 162 |
+
public $frames = array();
|
| 163 |
+
/**
|
| 164 |
+
* Document root, by default equals to document itself.
|
| 165 |
+
* Used by documentFragments.
|
| 166 |
+
*
|
| 167 |
+
* @var DOMNode
|
| 168 |
+
*/
|
| 169 |
+
public $root;
|
| 170 |
+
public $isDocumentFragment;
|
| 171 |
+
public $isXML = false;
|
| 172 |
+
public $isXHTML = false;
|
| 173 |
+
public $isHTML = false;
|
| 174 |
+
public $charset;
|
| 175 |
+
public function __construct($markup = null, $contentType = null, $newDocumentID = null) {
|
| 176 |
+
if (isset($markup))
|
| 177 |
+
$this->load($markup, $contentType, $newDocumentID);
|
| 178 |
+
$this->id = $newDocumentID
|
| 179 |
+
? $newDocumentID
|
| 180 |
+
: md5(microtime());
|
| 181 |
+
}
|
| 182 |
+
public function load($markup, $contentType = null, $newDocumentID = null) {
|
| 183 |
+
// phpQuery::$documents[$id] = $this;
|
| 184 |
+
$this->contentType = strtolower($contentType);
|
| 185 |
+
if ($markup instanceof DOMDOCUMENT) {
|
| 186 |
+
$this->document = $markup;
|
| 187 |
+
$this->root = $this->document;
|
| 188 |
+
$this->charset = $this->document->encoding;
|
| 189 |
+
// TODO isDocumentFragment
|
| 190 |
+
} else {
|
| 191 |
+
$loaded = $this->loadMarkup($markup);
|
| 192 |
+
}
|
| 193 |
+
if ($loaded) {
|
| 194 |
+
// $this->document->formatOutput = true;
|
| 195 |
+
$this->document->preserveWhiteSpace = true;
|
| 196 |
+
$this->xpath = new DOMXPath($this->document);
|
| 197 |
+
$this->afterMarkupLoad();
|
| 198 |
+
return true;
|
| 199 |
+
// remember last loaded document
|
| 200 |
+
// return phpQuery::selectDocument($id);
|
| 201 |
+
}
|
| 202 |
+
return false;
|
| 203 |
+
}
|
| 204 |
+
protected function afterMarkupLoad() {
|
| 205 |
+
if ($this->isXHTML) {
|
| 206 |
+
$this->xpath->registerNamespace("html", "http://www.w3.org/1999/xhtml");
|
| 207 |
+
}
|
| 208 |
+
}
|
| 209 |
+
protected function loadMarkup($markup) {
|
| 210 |
+
$loaded = false;
|
| 211 |
+
if ($this->contentType) {
|
| 212 |
+
self::debug("Load markup for content type {$this->contentType}");
|
| 213 |
+
// content determined by contentType
|
| 214 |
+
list($contentType, $charset) = $this->contentTypeToArray($this->contentType);
|
| 215 |
+
switch($contentType) {
|
| 216 |
+
case 'text/html':
|
| 217 |
+
phpQuery::debug("Loading HTML, content type '{$this->contentType}'");
|
| 218 |
+
$loaded = $this->loadMarkupHTML($markup, $charset);
|
| 219 |
+
break;
|
| 220 |
+
case 'text/xml':
|
| 221 |
+
case 'application/xhtml+xml':
|
| 222 |
+
phpQuery::debug("Loading XML, content type '{$this->contentType}'");
|
| 223 |
+
$loaded = $this->loadMarkupXML($markup, $charset);
|
| 224 |
+
break;
|
| 225 |
+
default:
|
| 226 |
+
// for feeds or anything that sometimes doesn't use text/xml
|
| 227 |
+
if (strpos('xml', $this->contentType) !== false) {
|
| 228 |
+
phpQuery::debug("Loading XML, content type '{$this->contentType}'");
|
| 229 |
+
$loaded = $this->loadMarkupXML($markup, $charset);
|
| 230 |
+
} else
|
| 231 |
+
phpQuery::debug("Could not determine document type from content type '{$this->contentType}'");
|
| 232 |
+
}
|
| 233 |
+
} else {
|
| 234 |
+
// content type autodetection
|
| 235 |
+
if ($this->isXML($markup)) {
|
| 236 |
+
phpQuery::debug("Loading XML, isXML() == true");
|
| 237 |
+
$loaded = $this->loadMarkupXML($markup);
|
| 238 |
+
if (! $loaded && $this->isXHTML) {
|
| 239 |
+
phpQuery::debug('Loading as XML failed, trying to load as HTML, isXHTML == true');
|
| 240 |
+
$loaded = $this->loadMarkupHTML($markup);
|
| 241 |
+
}
|
| 242 |
+
} else {
|
| 243 |
+
phpQuery::debug("Loading HTML, isXML() == false");
|
| 244 |
+
$loaded = $this->loadMarkupHTML($markup);
|
| 245 |
+
}
|
| 246 |
+
}
|
| 247 |
+
return $loaded;
|
| 248 |
+
}
|
| 249 |
+
protected function loadMarkupReset() {
|
| 250 |
+
$this->isXML = $this->isXHTML = $this->isHTML = false;
|
| 251 |
+
}
|
| 252 |
+
protected function documentCreate($charset, $version = '1.0') {
|
| 253 |
+
if (! $version)
|
| 254 |
+
$version = '1.0';
|
| 255 |
+
$this->document = new DOMDocument($version, $charset);
|
| 256 |
+
$this->charset = $this->document->encoding;
|
| 257 |
+
// $this->document->encoding = $charset;
|
| 258 |
+
$this->document->formatOutput = true;
|
| 259 |
+
$this->document->preserveWhiteSpace = true;
|
| 260 |
+
}
|
| 261 |
+
protected function loadMarkupHTML($markup, $requestedCharset = null) {
|
| 262 |
+
if (phpQuery::$debug)
|
| 263 |
+
phpQuery::debug('Full markup load (HTML): '.substr($markup, 0, 250));
|
| 264 |
+
$this->loadMarkupReset();
|
| 265 |
+
$this->isHTML = true;
|
| 266 |
+
if (!isset($this->isDocumentFragment))
|
| 267 |
+
$this->isDocumentFragment = self::isDocumentFragmentHTML($markup);
|
| 268 |
+
$charset = null;
|
| 269 |
+
$documentCharset = $this->charsetFromHTML($markup);
|
| 270 |
+
$addDocumentCharset = false;
|
| 271 |
+
if ($documentCharset) {
|
| 272 |
+
$charset = $documentCharset;
|
| 273 |
+
$markup = $this->charsetFixHTML($markup);
|
| 274 |
+
} else if ($requestedCharset) {
|
| 275 |
+
$charset = $requestedCharset;
|
| 276 |
+
}
|
| 277 |
+
if (! $charset)
|
| 278 |
+
$charset = phpQuery::$defaultCharset;
|
| 279 |
+
// HTTP 1.1 says that the default charset is ISO-8859-1
|
| 280 |
+
// @see http://www.w3.org/International/O-HTTP-charset
|
| 281 |
+
if (! $documentCharset) {
|
| 282 |
+
$documentCharset = 'ISO-8859-1';
|
| 283 |
+
$addDocumentCharset = true;
|
| 284 |
+
}
|
| 285 |
+
// Should be careful here, still need 'magic encoding detection' since lots of pages have other 'default encoding'
|
| 286 |
+
// Worse, some pages can have mixed encodings... we'll try not to worry about that
|
| 287 |
+
$requestedCharset = strtoupper($requestedCharset);
|
| 288 |
+
$documentCharset = strtoupper($documentCharset);
|
| 289 |
+
phpQuery::debug("DOC: $documentCharset REQ: $requestedCharset");
|
| 290 |
+
if ($requestedCharset && $documentCharset && $requestedCharset !== $documentCharset) {
|
| 291 |
+
phpQuery::debug("CHARSET CONVERT");
|
| 292 |
+
// Document Encoding Conversion
|
| 293 |
+
// http://code.google.com/p/phpquery/issues/detail?id=86
|
| 294 |
+
if (function_exists('mb_detect_encoding')) {
|
| 295 |
+
$possibleCharsets = array($documentCharset, $requestedCharset, 'AUTO');
|
| 296 |
+
$docEncoding = mb_detect_encoding($markup, implode(', ', $possibleCharsets));
|
| 297 |
+
if (! $docEncoding)
|
| 298 |
+
$docEncoding = $documentCharset; // ok trust the document
|
| 299 |
+
phpQuery::debug("DETECTED '$docEncoding'");
|
| 300 |
+
// Detected does not match what document says...
|
| 301 |
+
if ($docEncoding !== $documentCharset) {
|
| 302 |
+
// Tricky..
|
| 303 |
+
}
|
| 304 |
+
if ($docEncoding !== $requestedCharset) {
|
| 305 |
+
phpQuery::debug("CONVERT $docEncoding => $requestedCharset");
|
| 306 |
+
$markup = mb_convert_encoding($markup, $requestedCharset, $docEncoding);
|
| 307 |
+
$markup = $this->charsetAppendToHTML($markup, $requestedCharset);
|
| 308 |
+
$charset = $requestedCharset;
|
| 309 |
+
}
|
| 310 |
+
} else {
|
| 311 |
+
phpQuery::debug("TODO: charset conversion without mbstring...");
|
| 312 |
+
}
|
| 313 |
+
}
|
| 314 |
+
$return = false;
|
| 315 |
+
if ($this->isDocumentFragment) {
|
| 316 |
+
phpQuery::debug("Full markup load (HTML), DocumentFragment detected, using charset '$charset'");
|
| 317 |
+
$return = $this->documentFragmentLoadMarkup($this, $charset, $markup);
|
| 318 |
+
} else {
|
| 319 |
+
if ($addDocumentCharset) {
|
| 320 |
+
phpQuery::debug("Full markup load (HTML), appending charset: '$charset'");
|
| 321 |
+
$markup = $this->charsetAppendToHTML($markup, $charset);
|
| 322 |
+
}
|
| 323 |
+
phpQuery::debug("Full markup load (HTML), documentCreate('$charset')");
|
| 324 |
+
$this->documentCreate($charset);
|
| 325 |
+
$return = phpQuery::$debug === 2
|
| 326 |
+
? $this->document->loadHTML($markup)
|
| 327 |
+
: @$this->document->loadHTML($markup);
|
| 328 |
+
if ($return)
|
| 329 |
+
$this->root = $this->document;
|
| 330 |
+
}
|
| 331 |
+
if ($return && ! $this->contentType)
|
| 332 |
+
$this->contentType = 'text/html';
|
| 333 |
+
return $return;
|
| 334 |
+
}
|
| 335 |
+
protected function loadMarkupXML($markup, $requestedCharset = null) {
|
| 336 |
+
if (phpQuery::$debug)
|
| 337 |
+
phpQuery::debug('Full markup load (XML): '.substr($markup, 0, 250));
|
| 338 |
+
$this->loadMarkupReset();
|
| 339 |
+
$this->isXML = true;
|
| 340 |
+
// check agains XHTML in contentType or markup
|
| 341 |
+
$isContentTypeXHTML = $this->isXHTML();
|
| 342 |
+
$isMarkupXHTML = $this->isXHTML($markup);
|
| 343 |
+
if ($isContentTypeXHTML || $isMarkupXHTML) {
|
| 344 |
+
self::debug('Full markup load (XML), XHTML detected');
|
| 345 |
+
$this->isXHTML = true;
|
| 346 |
+
}
|
| 347 |
+
// determine document fragment
|
| 348 |
+
if (! isset($this->isDocumentFragment))
|
| 349 |
+
$this->isDocumentFragment = $this->isXHTML
|
| 350 |
+
? self::isDocumentFragmentXHTML($markup)
|
| 351 |
+
: self::isDocumentFragmentXML($markup);
|
| 352 |
+
// this charset will be used
|
| 353 |
+
$charset = null;
|
| 354 |
+
// charset from XML declaration @var string
|
| 355 |
+
$documentCharset = $this->charsetFromXML($markup);
|
| 356 |
+
if (! $documentCharset) {
|
| 357 |
+
if ($this->isXHTML) {
|
| 358 |
+
// this is XHTML, try to get charset from content-type meta header
|
| 359 |
+
$documentCharset = $this->charsetFromHTML($markup);
|
| 360 |
+
if ($documentCharset) {
|
| 361 |
+
phpQuery::debug("Full markup load (XML), appending XHTML charset '$documentCharset'");
|
| 362 |
+
$this->charsetAppendToXML($markup, $documentCharset);
|
| 363 |
+
$charset = $documentCharset;
|
| 364 |
+
}
|
| 365 |
+
}
|
| 366 |
+
if (! $documentCharset) {
|
| 367 |
+
// if still no document charset...
|
| 368 |
+
$charset = $requestedCharset;
|
| 369 |
+
}
|
| 370 |
+
} else if ($requestedCharset) {
|
| 371 |
+
$charset = $requestedCharset;
|
| 372 |
+
}
|
| 373 |
+
if (! $charset) {
|
| 374 |
+
$charset = phpQuery::$defaultCharset;
|
| 375 |
+
}
|
| 376 |
+
if ($requestedCharset && $documentCharset && $requestedCharset != $documentCharset) {
|
| 377 |
+
// TODO place for charset conversion
|
| 378 |
+
// $charset = $requestedCharset;
|
| 379 |
+
}
|
| 380 |
+
$return = false;
|
| 381 |
+
if ($this->isDocumentFragment) {
|
| 382 |
+
phpQuery::debug("Full markup load (XML), DocumentFragment detected, using charset '$charset'");
|
| 383 |
+
$return = $this->documentFragmentLoadMarkup($this, $charset, $markup);
|
| 384 |
+
} else {
|
| 385 |
+
// FIXME ???
|
| 386 |
+
if ($isContentTypeXHTML && ! $isMarkupXHTML)
|
| 387 |
+
if (! $documentCharset) {
|
| 388 |
+
phpQuery::debug("Full markup load (XML), appending charset '$charset'");
|
| 389 |
+
$markup = $this->charsetAppendToXML($markup, $charset);
|
| 390 |
+
}
|
| 391 |
+
// see http://pl2.php.net/manual/en/book.dom.php#78929
|
| 392 |
+
// LIBXML_DTDLOAD (>= PHP 5.1)
|
| 393 |
+
// does XML ctalogues works with LIBXML_NONET
|
| 394 |
+
// $this->document->resolveExternals = true;
|
| 395 |
+
// TODO test LIBXML_COMPACT for performance improvement
|
| 396 |
+
// create document
|
| 397 |
+
$this->documentCreate($charset);
|
| 398 |
+
if (phpversion() < 5.1) {
|
| 399 |
+
$this->document->resolveExternals = true;
|
| 400 |
+
$return = phpQuery::$debug === 2
|
| 401 |
+
? $this->document->loadXML($markup)
|
| 402 |
+
: @$this->document->loadXML($markup);
|
| 403 |
+
} else {
|
| 404 |
+
/** @link http://pl2.php.net/manual/en/libxml.constants.php */
|
| 405 |
+
$libxmlStatic = phpQuery::$debug === 2
|
| 406 |
+
? LIBXML_DTDLOAD|LIBXML_DTDATTR|LIBXML_NONET
|
| 407 |
+
: LIBXML_DTDLOAD|LIBXML_DTDATTR|LIBXML_NONET|LIBXML_NOWARNING|LIBXML_NOERROR;
|
| 408 |
+
$return = $this->document->loadXML($markup, $libxmlStatic);
|
| 409 |
+
// if (! $return)
|
| 410 |
+
// $return = $this->document->loadHTML($markup);
|
| 411 |
+
}
|
| 412 |
+
if ($return)
|
| 413 |
+
$this->root = $this->document;
|
| 414 |
+
}
|
| 415 |
+
if ($return) {
|
| 416 |
+
if (! $this->contentType) {
|
| 417 |
+
if ($this->isXHTML)
|
| 418 |
+
$this->contentType = 'application/xhtml+xml';
|
| 419 |
+
else
|
| 420 |
+
$this->contentType = 'text/xml';
|
| 421 |
+
}
|
| 422 |
+
return $return;
|
| 423 |
+
} else {
|
| 424 |
+
throw new Exception("Error loading XML markup");
|
| 425 |
+
}
|
| 426 |
+
}
|
| 427 |
+
protected function isXHTML($markup = null) {
|
| 428 |
+
if (! isset($markup)) {
|
| 429 |
+
return strpos($this->contentType, 'xhtml') !== false;
|
| 430 |
+
}
|
| 431 |
+
// XXX ok ?
|
| 432 |
+
return strpos($markup, "<!DOCTYPE html") !== false;
|
| 433 |
+
// return stripos($doctype, 'xhtml') !== false;
|
| 434 |
+
// $doctype = isset($dom->doctype) && is_object($dom->doctype)
|
| 435 |
+
// ? $dom->doctype->publicId
|
| 436 |
+
// : self::$defaultDoctype;
|
| 437 |
+
}
|
| 438 |
+
protected function isXML($markup) {
|
| 439 |
+
// return strpos($markup, '<?xml') !== false && stripos($markup, 'xhtml') === false;
|
| 440 |
+
return strpos(substr($markup, 0, 100), '<'.'?xml') !== false;
|
| 441 |
+
}
|
| 442 |
+
protected function contentTypeToArray($contentType) {
|
| 443 |
+
$matches = explode(';', trim(strtolower($contentType)));
|
| 444 |
+
if (isset($matches[1])) {
|
| 445 |
+
$matches[1] = explode('=', $matches[1]);
|
| 446 |
+
// strip 'charset='
|
| 447 |
+
$matches[1] = isset($matches[1][1]) && trim($matches[1][1])
|
| 448 |
+
? $matches[1][1]
|
| 449 |
+
: $matches[1][0];
|
| 450 |
+
} else
|
| 451 |
+
$matches[1] = null;
|
| 452 |
+
return $matches;
|
| 453 |
+
}
|
| 454 |
+
/**
|
| 455 |
+
*
|
| 456 |
+
* @param $markup
|
| 457 |
+
* @return array contentType, charset
|
| 458 |
+
*/
|
| 459 |
+
protected function contentTypeFromHTML($markup) {
|
| 460 |
+
$matches = array();
|
| 461 |
+
// find meta tag
|
| 462 |
+
preg_match('@<meta[^>]+http-equiv\\s*=\\s*(["|\'])Content-Type\\1([^>]+?)>@i',
|
| 463 |
+
$markup, $matches
|
| 464 |
+
);
|
| 465 |
+
if (! isset($matches[0]))
|
| 466 |
+
return array(null, null);
|
| 467 |
+
// get attr 'content'
|
| 468 |
+
preg_match('@content\\s*=\\s*(["|\'])(.+?)\\1@', $matches[0], $matches);
|
| 469 |
+
if (! isset($matches[0]))
|
| 470 |
+
return array(null, null);
|
| 471 |
+
return $this->contentTypeToArray($matches[2]);
|
| 472 |
+
}
|
| 473 |
+
protected function charsetFromHTML($markup) {
|
| 474 |
+
$contentType = $this->contentTypeFromHTML($markup);
|
| 475 |
+
return $contentType[1];
|
| 476 |
+
}
|
| 477 |
+
protected function charsetFromXML($markup) {
|
| 478 |
+
$matches;
|
| 479 |
+
// find declaration
|
| 480 |
+
preg_match('@<'.'?xml[^>]+encoding\\s*=\\s*(["|\'])(.*?)\\1@i',
|
| 481 |
+
$markup, $matches
|
| 482 |
+
);
|
| 483 |
+
return isset($matches[2])
|
| 484 |
+
? strtolower($matches[2])
|
| 485 |
+
: null;
|
| 486 |
+
}
|
| 487 |
+
/**
|
| 488 |
+
* Repositions meta[type=charset] at the start of head. Bypasses DOMDocument bug.
|
| 489 |
+
*
|
| 490 |
+
* @link http://code.google.com/p/phpquery/issues/detail?id=80
|
| 491 |
+
* @param $html
|
| 492 |
+
*/
|
| 493 |
+
protected function charsetFixHTML($markup) {
|
| 494 |
+
$matches = array();
|
| 495 |
+
// find meta tag
|
| 496 |
+
preg_match('@\s*<meta[^>]+http-equiv\\s*=\\s*(["|\'])Content-Type\\1([^>]+?)>@i',
|
| 497 |
+
$markup, $matches, PREG_OFFSET_CAPTURE
|
| 498 |
+
);
|
| 499 |
+
if (! isset($matches[0]))
|
| 500 |
+
return;
|
| 501 |
+
$metaContentType = $matches[0][0];
|
| 502 |
+
$markup = substr($markup, 0, $matches[0][1])
|
| 503 |
+
.substr($markup, $matches[0][1]+strlen($metaContentType));
|
| 504 |
+
$headStart = stripos($markup, '<head>');
|
| 505 |
+
$markup = substr($markup, 0, $headStart+6).$metaContentType
|
| 506 |
+
.substr($markup, $headStart+6);
|
| 507 |
+
return $markup;
|
| 508 |
+
}
|
| 509 |
+
protected function charsetAppendToHTML($html, $charset, $xhtml = false) {
|
| 510 |
+
// remove existing meta[type=content-type]
|
| 511 |
+
$html = preg_replace('@\s*<meta[^>]+http-equiv\\s*=\\s*(["|\'])Content-Type\\1([^>]+?)>@i', '', $html);
|
| 512 |
+
$meta = '<meta http-equiv="Content-Type" content="text/html;charset='
|
| 513 |
+
.$charset.'" '
|
| 514 |
+
.($xhtml ? '/' : '')
|
| 515 |
+
.'>';
|
| 516 |
+
if (strpos($html, '<head') === false) {
|
| 517 |
+
if (strpos($hltml, '<html') === false) {
|
| 518 |
+
return $meta.$html;
|
| 519 |
+
} else {
|
| 520 |
+
return preg_replace(
|
| 521 |
+
'@<html(.*?)(?(?<!\?)>)@s',
|
| 522 |
+
"<html\\1><head>{$meta}</head>",
|
| 523 |
+
$html
|
| 524 |
+
);
|
| 525 |
+
}
|
| 526 |
+
} else {
|
| 527 |
+
return preg_replace(
|
| 528 |
+
'@<head(.*?)(?(?<!\?)>)@s',
|
| 529 |
+
'<head\\1>'.$meta,
|
| 530 |
+
$html
|
| 531 |
+
);
|
| 532 |
+
}
|
| 533 |
+
}
|
| 534 |
+
protected function charsetAppendToXML($markup, $charset) {
|
| 535 |
+
$declaration = '<'.'?xml version="1.0" encoding="'.$charset.'"?'.'>';
|
| 536 |
+
return $declaration.$markup;
|
| 537 |
+
}
|
| 538 |
+
public static function isDocumentFragmentHTML($markup) {
|
| 539 |
+
return stripos($markup, '<html') === false && stripos($markup, '<!doctype') === false;
|
| 540 |
+
}
|
| 541 |
+
public static function isDocumentFragmentXML($markup) {
|
| 542 |
+
return stripos($markup, '<'.'?xml') === false;
|
| 543 |
+
}
|
| 544 |
+
public static function isDocumentFragmentXHTML($markup) {
|
| 545 |
+
return self::isDocumentFragmentHTML($markup);
|
| 546 |
+
}
|
| 547 |
+
public function importAttr($value) {
|
| 548 |
+
// TODO
|
| 549 |
+
}
|
| 550 |
+
/**
|
| 551 |
+
*
|
| 552 |
+
* @param $source
|
| 553 |
+
* @param $target
|
| 554 |
+
* @param $sourceCharset
|
| 555 |
+
* @return array Array of imported nodes.
|
| 556 |
+
*/
|
| 557 |
+
public function import($source, $sourceCharset = null) {
|
| 558 |
+
// TODO charset conversions
|
| 559 |
+
$return = array();
|
| 560 |
+
if ($source instanceof DOMNODE && !($source instanceof DOMNODELIST))
|
| 561 |
+
$source = array($source);
|
| 562 |
+
// if (is_array($source)) {
|
| 563 |
+
// foreach($source as $node) {
|
| 564 |
+
// if (is_string($node)) {
|
| 565 |
+
// // string markup
|
| 566 |
+
// $fake = $this->documentFragmentCreate($node, $sourceCharset);
|
| 567 |
+
// if ($fake === false)
|
| 568 |
+
// throw new Exception("Error loading documentFragment markup");
|
| 569 |
+
// else
|
| 570 |
+
// $return = array_merge($return,
|
| 571 |
+
// $this->import($fake->root->childNodes)
|
| 572 |
+
// );
|
| 573 |
+
// } else {
|
| 574 |
+
// $return[] = $this->document->importNode($node, true);
|
| 575 |
+
// }
|
| 576 |
+
// }
|
| 577 |
+
// return $return;
|
| 578 |
+
// } else {
|
| 579 |
+
// // string markup
|
| 580 |
+
// $fake = $this->documentFragmentCreate($source, $sourceCharset);
|
| 581 |
+
// if ($fake === false)
|
| 582 |
+
// throw new Exception("Error loading documentFragment markup");
|
| 583 |
+
// else
|
| 584 |
+
// return $this->import($fake->root->childNodes);
|
| 585 |
+
// }
|
| 586 |
+
if (is_array($source) || $source instanceof DOMNODELIST) {
|
| 587 |
+
// dom nodes
|
| 588 |
+
self::debug('Importing nodes to document');
|
| 589 |
+
foreach($source as $node)
|
| 590 |
+
$return[] = $this->document->importNode($node, true);
|
| 591 |
+
} else {
|
| 592 |
+
// string markup
|
| 593 |
+
$fake = $this->documentFragmentCreate($source, $sourceCharset);
|
| 594 |
+
if ($fake === false)
|
| 595 |
+
throw new Exception("Error loading documentFragment markup");
|
| 596 |
+
else
|
| 597 |
+
return $this->import($fake->root->childNodes);
|
| 598 |
+
}
|
| 599 |
+
return $return;
|
| 600 |
+
}
|
| 601 |
+
/**
|
| 602 |
+
* Creates new document fragment.
|
| 603 |
+
*
|
| 604 |
+
* @param $source
|
| 605 |
+
* @return DOMDocumentWrapper
|
| 606 |
+
*/
|
| 607 |
+
protected function documentFragmentCreate($source, $charset = null) {
|
| 608 |
+
$fake = new DOMDocumentWrapper();
|
| 609 |
+
$fake->contentType = $this->contentType;
|
| 610 |
+
$fake->isXML = $this->isXML;
|
| 611 |
+
$fake->isHTML = $this->isHTML;
|
| 612 |
+
$fake->isXHTML = $this->isXHTML;
|
| 613 |
+
$fake->root = $fake->document;
|
| 614 |
+
if (! $charset)
|
| 615 |
+
$charset = $this->charset;
|
| 616 |
+
// $fake->documentCreate($this->charset);
|
| 617 |
+
if ($source instanceof DOMNODE && !($source instanceof DOMNODELIST))
|
| 618 |
+
$source = array($source);
|
| 619 |
+
if (is_array($source) || $source instanceof DOMNODELIST) {
|
| 620 |
+
// dom nodes
|
| 621 |
+
// load fake document
|
| 622 |
+
if (! $this->documentFragmentLoadMarkup($fake, $charset))
|
| 623 |
+
return false;
|
| 624 |
+
$nodes = $fake->import($source);
|
| 625 |
+
foreach($nodes as $node)
|
| 626 |
+
$fake->root->appendChild($node);
|
| 627 |
+
} else {
|
| 628 |
+
// string markup
|
| 629 |
+
$this->documentFragmentLoadMarkup($fake, $charset, $source);
|
| 630 |
+
}
|
| 631 |
+
return $fake;
|
| 632 |
+
}
|
| 633 |
+
/**
|
| 634 |
+
*
|
| 635 |
+
* @param $document DOMDocumentWrapper
|
| 636 |
+
* @param $markup
|
| 637 |
+
* @return $document
|
| 638 |
+
*/
|
| 639 |
+
private function documentFragmentLoadMarkup($fragment, $charset, $markup = null) {
|
| 640 |
+
// TODO error handling
|
| 641 |
+
// TODO copy doctype
|
| 642 |
+
// tempolary turn off
|
| 643 |
+
$fragment->isDocumentFragment = false;
|
| 644 |
+
if ($fragment->isXML) {
|
| 645 |
+
if ($fragment->isXHTML) {
|
| 646 |
+
// add FAKE element to set default namespace
|
| 647 |
+
$fragment->loadMarkupXML('<?xml version="1.0" encoding="'.$charset.'"?>'
|
| 648 |
+
.'<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" '
|
| 649 |
+
.'"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">'
|
| 650 |
+
.'<fake xmlns="http://www.w3.org/1999/xhtml">'.$markup.'</fake>');
|
| 651 |
+
$fragment->root = $fragment->document->firstChild->nextSibling;
|
| 652 |
+
} else {
|
| 653 |
+
$fragment->loadMarkupXML('<?xml version="1.0" encoding="'.$charset.'"?><fake>'.$markup.'</fake>');
|
| 654 |
+
$fragment->root = $fragment->document->firstChild;
|
| 655 |
+
}
|
| 656 |
+
} else {
|
| 657 |
+
$markup2 = phpQuery::$defaultDoctype.'<html><head><meta http-equiv="Content-Type" content="text/html;charset='
|
| 658 |
+
.$charset.'"></head>';
|
| 659 |
+
$noBody = strpos($markup, '<body') === false;
|
| 660 |
+
if ($noBody)
|
| 661 |
+
$markup2 .= '<body>';
|
| 662 |
+
$markup2 .= $markup;
|
| 663 |
+
if ($noBody)
|
| 664 |
+
$markup2 .= '</body>';
|
| 665 |
+
$markup2 .= '</html>';
|
| 666 |
+
$fragment->loadMarkupHTML($markup2);
|
| 667 |
+
// TODO resolv body tag merging issue
|
| 668 |
+
$fragment->root = $noBody
|
| 669 |
+
? $fragment->document->firstChild->nextSibling->firstChild->nextSibling
|
| 670 |
+
: $fragment->document->firstChild->nextSibling->firstChild->nextSibling;
|
| 671 |
+
}
|
| 672 |
+
if (! $fragment->root)
|
| 673 |
+
return false;
|
| 674 |
+
$fragment->isDocumentFragment = true;
|
| 675 |
+
return true;
|
| 676 |
+
}
|
| 677 |
+
protected function documentFragmentToMarkup($fragment) {
|
| 678 |
+
phpQuery::debug('documentFragmentToMarkup');
|
| 679 |
+
$tmp = $fragment->isDocumentFragment;
|
| 680 |
+
$fragment->isDocumentFragment = false;
|
| 681 |
+
$markup = $fragment->markup();
|
| 682 |
+
if ($fragment->isXML) {
|
| 683 |
+
$markup = substr($markup, 0, strrpos($markup, '</fake>'));
|
| 684 |
+
if ($fragment->isXHTML) {
|
| 685 |
+
$markup = substr($markup, strpos($markup, '<fake')+43);
|
| 686 |
+
} else {
|
| 687 |
+
$markup = substr($markup, strpos($markup, '<fake>')+6);
|
| 688 |
+
}
|
| 689 |
+
} else {
|
| 690 |
+
$markup = substr($markup, strpos($markup, '<body>')+6);
|
| 691 |
+
$markup = substr($markup, 0, strrpos($markup, '</body>'));
|
| 692 |
+
}
|
| 693 |
+
$fragment->isDocumentFragment = $tmp;
|
| 694 |
+
if (phpQuery::$debug)
|
| 695 |
+
phpQuery::debug('documentFragmentToMarkup: '.substr($markup, 0, 150));
|
| 696 |
+
return $markup;
|
| 697 |
+
}
|
| 698 |
+
/**
|
| 699 |
+
* Return document markup, starting with optional $nodes as root.
|
| 700 |
+
*
|
| 701 |
+
* @param $nodes DOMNode|DOMNodeList
|
| 702 |
+
* @return string
|
| 703 |
+
*/
|
| 704 |
+
public function markup($nodes = null, $innerMarkup = false) {
|
| 705 |
+
if (isset($nodes) && count($nodes) == 1 && $nodes[0] instanceof DOMDOCUMENT)
|
| 706 |
+
$nodes = null;
|
| 707 |
+
if (isset($nodes)) {
|
| 708 |
+
$markup = '';
|
| 709 |
+
if (!is_array($nodes) && !($nodes instanceof DOMNODELIST) )
|
| 710 |
+
$nodes = array($nodes);
|
| 711 |
+
if ($this->isDocumentFragment && ! $innerMarkup)
|
| 712 |
+
foreach($nodes as $i => $node)
|
| 713 |
+
if ($node->isSameNode($this->root)) {
|
| 714 |
+
// var_dump($node);
|
| 715 |
+
$nodes = array_slice($nodes, 0, $i)
|
| 716 |
+
+ phpQuery::DOMNodeListToArray($node->childNodes)
|
| 717 |
+
+ array_slice($nodes, $i+1);
|
| 718 |
+
}
|
| 719 |
+
if ($this->isXML && ! $innerMarkup) {
|
| 720 |
+
self::debug("Getting outerXML with charset '{$this->charset}'");
|
| 721 |
+
// we need outerXML, so we can benefit from
|
| 722 |
+
// $node param support in saveXML()
|
| 723 |
+
foreach($nodes as $node)
|
| 724 |
+
$markup .= $this->document->saveXML($node);
|
| 725 |
+
} else {
|
| 726 |
+
$loop = array();
|
| 727 |
+
if ($innerMarkup)
|
| 728 |
+
foreach($nodes as $node) {
|
| 729 |
+
if ($node->childNodes)
|
| 730 |
+
foreach($node->childNodes as $child)
|
| 731 |
+
$loop[] = $child;
|
| 732 |
+
else
|
| 733 |
+
$loop[] = $node;
|
| 734 |
+
}
|
| 735 |
+
else
|
| 736 |
+
$loop = $nodes;
|
| 737 |
+
self::debug("Getting markup, moving selected nodes (".count($loop).") to new DocumentFragment");
|
| 738 |
+
$fake = $this->documentFragmentCreate($loop);
|
| 739 |
+
$markup = $this->documentFragmentToMarkup($fake);
|
| 740 |
+
}
|
| 741 |
+
if ($this->isXHTML) {
|
| 742 |
+
self::debug("Fixing XHTML");
|
| 743 |
+
$markup = self::markupFixXHTML($markup);
|
| 744 |
+
}
|
| 745 |
+
self::debug("Markup: ".substr($markup, 0, 250));
|
| 746 |
+
return $markup;
|
| 747 |
+
} else {
|
| 748 |
+
if ($this->isDocumentFragment) {
|
| 749 |
+
// documentFragment, html only...
|
| 750 |
+
self::debug("Getting markup, DocumentFragment detected");
|
| 751 |
+
// return $this->markup(
|
| 752 |
+
//// $this->document->getElementsByTagName('body')->item(0)
|
| 753 |
+
// $this->document->root, true
|
| 754 |
+
// );
|
| 755 |
+
$markup = $this->documentFragmentToMarkup($this);
|
| 756 |
+
// no need for markupFixXHTML, as it's done thought markup($nodes) method
|
| 757 |
+
return $markup;
|
| 758 |
+
} else {
|
| 759 |
+
self::debug("Getting markup (".($this->isXML?'XML':'HTML')."), final with charset '{$this->charset}'");
|
| 760 |
+
$markup = $this->isXML
|
| 761 |
+
? $this->document->saveXML()
|
| 762 |
+
: $this->document->saveHTML();
|
| 763 |
+
if ($this->isXHTML) {
|
| 764 |
+
self::debug("Fixing XHTML");
|
| 765 |
+
$markup = self::markupFixXHTML($markup);
|
| 766 |
+
}
|
| 767 |
+
self::debug("Markup: ".substr($markup, 0, 250));
|
| 768 |
+
return $markup;
|
| 769 |
+
}
|
| 770 |
+
}
|
| 771 |
+
}
|
| 772 |
+
protected static function markupFixXHTML($markup) {
|
| 773 |
+
$markup = self::expandEmptyTag('script', $markup);
|
| 774 |
+
$markup = self::expandEmptyTag('select', $markup);
|
| 775 |
+
$markup = self::expandEmptyTag('textarea', $markup);
|
| 776 |
+
return $markup;
|
| 777 |
+
}
|
| 778 |
+
public static function debug($text) {
|
| 779 |
+
phpQuery::debug($text);
|
| 780 |
+
}
|
| 781 |
+
/**
|
| 782 |
+
* expandEmptyTag
|
| 783 |
+
*
|
| 784 |
+
* @param $tag
|
| 785 |
+
* @param $xml
|
| 786 |
+
* @return unknown_type
|
| 787 |
+
* @author mjaque at ilkebenson dot com
|
| 788 |
+
* @link http://php.net/manual/en/domdocument.savehtml.php#81256
|
| 789 |
+
*/
|
| 790 |
+
public static function expandEmptyTag($tag, $xml){
|
| 791 |
+
$indice = 0;
|
| 792 |
+
while ($indice< strlen($xml)){
|
| 793 |
+
$pos = strpos($xml, "<$tag ", $indice);
|
| 794 |
+
if ($pos){
|
| 795 |
+
$posCierre = strpos($xml, ">", $pos);
|
| 796 |
+
if ($xml[$posCierre-1] == "/"){
|
| 797 |
+
$xml = substr_replace($xml, "></$tag>", $posCierre-1, 2);
|
| 798 |
+
}
|
| 799 |
+
$indice = $posCierre;
|
| 800 |
+
}
|
| 801 |
+
else break;
|
| 802 |
+
}
|
| 803 |
+
return $xml;
|
| 804 |
+
}
|
| 805 |
+
}
|
| 806 |
+
|
| 807 |
+
/**
|
| 808 |
+
* Event handling class.
|
| 809 |
+
*
|
| 810 |
+
* @author Tobiasz Cudnik
|
| 811 |
+
* @package phpQuery
|
| 812 |
+
* @static
|
| 813 |
+
*/
|
| 814 |
+
abstract class phpQueryEvents {
|
| 815 |
+
/**
|
| 816 |
+
* Trigger a type of event on every matched element.
|
| 817 |
+
*
|
| 818 |
+
* @param DOMNode|phpQueryObject|string $document
|
| 819 |
+
* @param unknown_type $type
|
| 820 |
+
* @param unknown_type $data
|
| 821 |
+
*
|
| 822 |
+
* @TODO exclusive events (with !)
|
| 823 |
+
* @TODO global events (test)
|
| 824 |
+
* @TODO support more than event in $type (space-separated)
|
| 825 |
+
*/
|
| 826 |
+
public static function trigger($document, $type, $data = array(), $node = null) {
|
| 827 |
+
// trigger: function(type, data, elem, donative, extra) {
|
| 828 |
+
$documentID = phpQuery::getDocumentID($document);
|
| 829 |
+
$namespace = null;
|
| 830 |
+
if (strpos($type, '.') !== false)
|
| 831 |
+
list($name, $namespace) = explode('.', $type);
|
| 832 |
+
else
|
| 833 |
+
$name = $type;
|
| 834 |
+
if (! $node) {
|
| 835 |
+
if (self::issetGlobal($documentID, $type)) {
|
| 836 |
+
$pq = phpQuery::getDocument($documentID);
|
| 837 |
+
// TODO check add($pq->document)
|
| 838 |
+
$pq->find('*')->add($pq->document)
|
| 839 |
+
->trigger($type, $data);
|
| 840 |
+
}
|
| 841 |
+
} else {
|
| 842 |
+
if (isset($data[0]) && $data[0] instanceof DOMEvent) {
|
| 843 |
+
$event = $data[0];
|
| 844 |
+
$event->relatedTarget = $event->target;
|
| 845 |
+
$event->target = $node;
|
| 846 |
+
$data = array_slice($data, 1);
|
| 847 |
+
} else {
|
| 848 |
+
$event = new DOMEvent(array(
|
| 849 |
+
'type' => $type,
|
| 850 |
+
'target' => $node,
|
| 851 |
+
'timeStamp' => time(),
|
| 852 |
+
));
|
| 853 |
+
}
|
| 854 |
+
$i = 0;
|
| 855 |
+
while($node) {
|
| 856 |
+
// TODO whois
|
| 857 |
+
phpQuery::debug("Triggering ".($i?"bubbled ":'')."event '{$type}' on "
|
| 858 |
+
."node \n");//.phpQueryObject::whois($node)."\n");
|
| 859 |
+
$event->currentTarget = $node;
|
| 860 |
+
$eventNode = self::getNode($documentID, $node);
|
| 861 |
+
if (isset($eventNode->eventHandlers)) {
|
| 862 |
+
foreach($eventNode->eventHandlers as $eventType => $handlers) {
|
| 863 |
+
$eventNamespace = null;
|
| 864 |
+
if (strpos($type, '.') !== false)
|
| 865 |
+
list($eventName, $eventNamespace) = explode('.', $eventType);
|
| 866 |
+
else
|
| 867 |
+
$eventName = $eventType;
|
| 868 |
+
if ($name != $eventName)
|
| 869 |
+
continue;
|
| 870 |
+
if ($namespace && $eventNamespace && $namespace != $eventNamespace)
|
| 871 |
+
continue;
|
| 872 |
+
foreach($handlers as $handler) {
|
| 873 |
+
phpQuery::debug("Calling event handler\n");
|
| 874 |
+
$event->data = $handler['data']
|
| 875 |
+
? $handler['data']
|
| 876 |
+
: null;
|
| 877 |
+
$params = array_merge(array($event), $data);
|
| 878 |
+
$return = phpQuery::callbackRun($handler['callback'], $params);
|
| 879 |
+
if ($return === false) {
|
| 880 |
+
$event->bubbles = false;
|
| 881 |
+
}
|
| 882 |
+
}
|
| 883 |
+
}
|
| 884 |
+
}
|
| 885 |
+
// to bubble or not to bubble...
|
| 886 |
+
if (! $event->bubbles)
|
| 887 |
+
break;
|
| 888 |
+
$node = $node->parentNode;
|
| 889 |
+
$i++;
|
| 890 |
+
}
|
| 891 |
+
}
|
| 892 |
+
}
|
| 893 |
+
/**
|
| 894 |
+
* Binds a handler to one or more events (like click) for each matched element.
|
| 895 |
+
* Can also bind custom events.
|
| 896 |
+
*
|
| 897 |
+
* @param DOMNode|phpQueryObject|string $document
|
| 898 |
+
* @param unknown_type $type
|
| 899 |
+
* @param unknown_type $data Optional
|
| 900 |
+
* @param unknown_type $callback
|
| 901 |
+
*
|
| 902 |
+
* @TODO support '!' (exclusive) events
|
| 903 |
+
* @TODO support more than event in $type (space-separated)
|
| 904 |
+
* @TODO support binding to global events
|
| 905 |
+
*/
|
| 906 |
+
public static function add($document, $node, $type, $data, $callback = null) {
|
| 907 |
+
phpQuery::debug("Binding '$type' event");
|
| 908 |
+
$documentID = phpQuery::getDocumentID($document);
|
| 909 |
+
// if (is_null($callback) && is_callable($data)) {
|
| 910 |
+
// $callback = $data;
|
| 911 |
+
// $data = null;
|
| 912 |
+
// }
|
| 913 |
+
$eventNode = self::getNode($documentID, $node);
|
| 914 |
+
if (! $eventNode)
|
| 915 |
+
$eventNode = self::setNode($documentID, $node);
|
| 916 |
+
if (!isset($eventNode->eventHandlers[$type]))
|
| 917 |
+
$eventNode->eventHandlers[$type] = array();
|
| 918 |
+
$eventNode->eventHandlers[$type][] = array(
|
| 919 |
+
'callback' => $callback,
|
| 920 |
+
'data' => $data,
|
| 921 |
+
);
|
| 922 |
+
}
|
| 923 |
+
/**
|
| 924 |
+
* Enter description here...
|
| 925 |
+
*
|
| 926 |
+
* @param DOMNode|phpQueryObject|string $document
|
| 927 |
+
* @param unknown_type $type
|
| 928 |
+
* @param unknown_type $callback
|
| 929 |
+
*
|
| 930 |
+
* @TODO namespace events
|
| 931 |
+
* @TODO support more than event in $type (space-separated)
|
| 932 |
+
*/
|
| 933 |
+
public static function remove($document, $node, $type = null, $callback = null) {
|
| 934 |
+
$documentID = phpQuery::getDocumentID($document);
|
| 935 |
+
$eventNode = self::getNode($documentID, $node);
|
| 936 |
+
if (is_object($eventNode) && isset($eventNode->eventHandlers[$type])) {
|
| 937 |
+
if ($callback) {
|
| 938 |
+
foreach($eventNode->eventHandlers[$type] as $k => $handler)
|
| 939 |
+
if ($handler['callback'] == $callback)
|
| 940 |
+
unset($eventNode->eventHandlers[$type][$k]);
|
| 941 |
+
} else {
|
| 942 |
+
unset($eventNode->eventHandlers[$type]);
|
| 943 |
+
}
|
| 944 |
+
}
|
| 945 |
+
}
|
| 946 |
+
protected static function getNode($documentID, $node) {
|
| 947 |
+
foreach(phpQuery::$documents[$documentID]->eventsNodes as $eventNode) {
|
| 948 |
+
if ($node->isSameNode($eventNode))
|
| 949 |
+
return $eventNode;
|
| 950 |
+
}
|
| 951 |
+
}
|
| 952 |
+
protected static function setNode($documentID, $node) {
|
| 953 |
+
phpQuery::$documents[$documentID]->eventsNodes[] = $node;
|
| 954 |
+
return phpQuery::$documents[$documentID]->eventsNodes[
|
| 955 |
+
count(phpQuery::$documents[$documentID]->eventsNodes)-1
|
| 956 |
+
];
|
| 957 |
+
}
|
| 958 |
+
protected static function issetGlobal($documentID, $type) {
|
| 959 |
+
return isset(phpQuery::$documents[$documentID])
|
| 960 |
+
? in_array($type, phpQuery::$documents[$documentID]->eventsGlobal)
|
| 961 |
+
: false;
|
| 962 |
+
}
|
| 963 |
+
}
|
| 964 |
+
|
| 965 |
+
|
| 966 |
+
interface ICallbackNamed {
|
| 967 |
+
function hasName();
|
| 968 |
+
function getName();
|
| 969 |
+
}
|
| 970 |
+
/**
|
| 971 |
+
* Callback class introduces currying-like pattern.
|
| 972 |
+
*
|
| 973 |
+
* Example:
|
| 974 |
+
* function foo($param1, $param2, $param3) {
|
| 975 |
+
* var_dump($param1, $param2, $param3);
|
| 976 |
+
* }
|
| 977 |
+
* $fooCurried = new Callback('foo',
|
| 978 |
+
* 'param1 is now statically set',
|
| 979 |
+
* new CallbackParam, new CallbackParam
|
| 980 |
+
* );
|
| 981 |
+
* phpQuery::callbackRun($fooCurried,
|
| 982 |
+
* array('param2 value', 'param3 value'
|
| 983 |
+
* );
|
| 984 |
+
*
|
| 985 |
+
* Callback class is supported in all phpQuery methods which accepts callbacks.
|
| 986 |
+
*
|
| 987 |
+
* @link http://code.google.com/p/phpquery/wiki/Callbacks#Param_Structures
|
| 988 |
+
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 989 |
+
*
|
| 990 |
+
* @TODO??? return fake forwarding function created via create_function
|
| 991 |
+
* @TODO honor paramStructure
|
| 992 |
+
*/
|
| 993 |
+
class Callback
|
| 994 |
+
implements ICallbackNamed {
|
| 995 |
+
public $callback = null;
|
| 996 |
+
public $params = null;
|
| 997 |
+
protected $name;
|
| 998 |
+
public function __construct($callback, $param1 = null, $param2 = null,
|
| 999 |
+
$param3 = null) {
|
| 1000 |
+
$params = func_get_args();
|
| 1001 |
+
$params = array_slice($params, 1);
|
| 1002 |
+
if ($callback instanceof Callback) {
|
| 1003 |
+
// TODO implement recurention
|
| 1004 |
+
} else {
|
| 1005 |
+
$this->callback = $callback;
|
| 1006 |
+
$this->params = $params;
|
| 1007 |
+
}
|
| 1008 |
+
}
|
| 1009 |
+
public function getName() {
|
| 1010 |
+
return 'Callback: '.$this->name;
|
| 1011 |
+
}
|
| 1012 |
+
public function hasName() {
|
| 1013 |
+
return isset($this->name) && $this->name;
|
| 1014 |
+
}
|
| 1015 |
+
public function setName($name) {
|
| 1016 |
+
$this->name = $name;
|
| 1017 |
+
return $this;
|
| 1018 |
+
}
|
| 1019 |
+
// TODO test me
|
| 1020 |
+
// public function addParams() {
|
| 1021 |
+
// $params = func_get_args();
|
| 1022 |
+
// return new Callback($this->callback, $this->params+$params);
|
| 1023 |
+
// }
|
| 1024 |
+
}
|
| 1025 |
+
/**
|
| 1026 |
+
* Shorthand for new Callback(create_function(...), ...);
|
| 1027 |
+
*
|
| 1028 |
+
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 1029 |
+
*/
|
| 1030 |
+
class CallbackBody extends Callback {
|
| 1031 |
+
public function __construct($paramList, $code, $param1 = null, $param2 = null,
|
| 1032 |
+
$param3 = null) {
|
| 1033 |
+
$params = func_get_args();
|
| 1034 |
+
$params = array_slice($params, 2);
|
| 1035 |
+
$this->callback = create_function($paramList, $code);
|
| 1036 |
+
$this->params = $params;
|
| 1037 |
+
}
|
| 1038 |
+
}
|
| 1039 |
+
/**
|
| 1040 |
+
* Callback type which on execution returns reference passed during creation.
|
| 1041 |
+
*
|
| 1042 |
+
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 1043 |
+
*/
|
| 1044 |
+
class CallbackReturnReference extends Callback
|
| 1045 |
+
implements ICallbackNamed {
|
| 1046 |
+
protected $reference;
|
| 1047 |
+
public function __construct(&$reference, $name = null){
|
| 1048 |
+
$this->reference =& $reference;
|
| 1049 |
+
$this->callback = array($this, 'callback');
|
| 1050 |
+
}
|
| 1051 |
+
public function callback() {
|
| 1052 |
+
return $this->reference;
|
| 1053 |
+
}
|
| 1054 |
+
public function getName() {
|
| 1055 |
+
return 'Callback: '.$this->name;
|
| 1056 |
+
}
|
| 1057 |
+
public function hasName() {
|
| 1058 |
+
return isset($this->name) && $this->name;
|
| 1059 |
+
}
|
| 1060 |
+
}
|
| 1061 |
+
/**
|
| 1062 |
+
* Callback type which on execution returns value passed during creation.
|
| 1063 |
+
*
|
| 1064 |
+
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 1065 |
+
*/
|
| 1066 |
+
class CallbackReturnValue extends Callback
|
| 1067 |
+
implements ICallbackNamed {
|
| 1068 |
+
protected $value;
|
| 1069 |
+
protected $name;
|
| 1070 |
+
public function __construct($value, $name = null){
|
| 1071 |
+
$this->value =& $value;
|
| 1072 |
+
$this->name = $name;
|
| 1073 |
+
$this->callback = array($this, 'callback');
|
| 1074 |
+
}
|
| 1075 |
+
public function callback() {
|
| 1076 |
+
return $this->value;
|
| 1077 |
+
}
|
| 1078 |
+
public function __toString() {
|
| 1079 |
+
return $this->getName();
|
| 1080 |
+
}
|
| 1081 |
+
public function getName() {
|
| 1082 |
+
return 'Callback: '.$this->name;
|
| 1083 |
+
}
|
| 1084 |
+
public function hasName() {
|
| 1085 |
+
return isset($this->name) && $this->name;
|
| 1086 |
+
}
|
| 1087 |
+
}
|
| 1088 |
+
/**
|
| 1089 |
+
* CallbackParameterToReference can be used when we don't really want a callback,
|
| 1090 |
+
* only parameter passed to it. CallbackParameterToReference takes first
|
| 1091 |
+
* parameter's value and passes it to reference.
|
| 1092 |
+
*
|
| 1093 |
+
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 1094 |
+
*/
|
| 1095 |
+
class CallbackParameterToReference extends Callback {
|
| 1096 |
+
/**
|
| 1097 |
+
* @param $reference
|
| 1098 |
+
* @TODO implement $paramIndex;
|
| 1099 |
+
* param index choose which callback param will be passed to reference
|
| 1100 |
+
*/
|
| 1101 |
+
public function __construct(&$reference){
|
| 1102 |
+
$this->callback =& $reference;
|
| 1103 |
+
}
|
| 1104 |
+
}
|
| 1105 |
+
//class CallbackReference extends Callback {
|
| 1106 |
+
// /**
|
| 1107 |
+
// *
|
| 1108 |
+
// * @param $reference
|
| 1109 |
+
// * @param $paramIndex
|
| 1110 |
+
// * @todo implement $paramIndex; param index choose which callback param will be passed to reference
|
| 1111 |
+
// */
|
| 1112 |
+
// public function __construct(&$reference, $name = null){
|
| 1113 |
+
// $this->callback =& $reference;
|
| 1114 |
+
// }
|
| 1115 |
+
//}
|
| 1116 |
+
class CallbackParam {}
|
| 1117 |
+
|
| 1118 |
+
/**
|
| 1119 |
+
* Class representing phpQuery objects.
|
| 1120 |
+
*
|
| 1121 |
+
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 1122 |
+
* @package phpQuery
|
| 1123 |
+
* @method phpQueryObject clone() clone()
|
| 1124 |
+
* @method phpQueryObject empty() empty()
|
| 1125 |
+
* @method phpQueryObject next() next($selector = null)
|
| 1126 |
+
* @method phpQueryObject prev() prev($selector = null)
|
| 1127 |
+
* @property Int $length
|
| 1128 |
+
*/
|
| 1129 |
+
class phpQueryObject
|
| 1130 |
+
implements Iterator, Countable, ArrayAccess {
|
| 1131 |
+
public $documentID = null;
|
| 1132 |
+
/**
|
| 1133 |
+
* DOMDocument class.
|
| 1134 |
+
*
|
| 1135 |
+
* @var DOMDocument
|
| 1136 |
+
*/
|
| 1137 |
+
public $document = null;
|
| 1138 |
+
public $charset = null;
|
| 1139 |
+
/**
|
| 1140 |
+
*
|
| 1141 |
+
* @var DOMDocumentWrapper
|
| 1142 |
+
*/
|
| 1143 |
+
public $documentWrapper = null;
|
| 1144 |
+
/**
|
| 1145 |
+
* XPath interface.
|
| 1146 |
+
*
|
| 1147 |
+
* @var DOMXPath
|
| 1148 |
+
*/
|
| 1149 |
+
public $xpath = null;
|
| 1150 |
+
/**
|
| 1151 |
+
* Stack of selected elements.
|
| 1152 |
+
* @TODO refactor to ->nodes
|
| 1153 |
+
* @var array
|
| 1154 |
+
*/
|
| 1155 |
+
public $elements = array();
|
| 1156 |
+
/**
|
| 1157 |
+
* @access private
|
| 1158 |
+
*/
|
| 1159 |
+
protected $elementsBackup = array();
|
| 1160 |
+
/**
|
| 1161 |
+
* @access private
|
| 1162 |
+
*/
|
| 1163 |
+
protected $previous = null;
|
| 1164 |
+
/**
|
| 1165 |
+
* @access private
|
| 1166 |
+
* @TODO deprecate
|
| 1167 |
+
*/
|
| 1168 |
+
protected $root = array();
|
| 1169 |
+
/**
|
| 1170 |
+
* Indicated if doument is just a fragment (no <html> tag).
|
| 1171 |
+
*
|
| 1172 |
+
* Every document is realy a full document, so even documentFragments can
|
| 1173 |
+
* be queried against <html>, but getDocument(id)->htmlOuter() will return
|
| 1174 |
+
* only contents of <body>.
|
| 1175 |
+
*
|
| 1176 |
+
* @var bool
|
| 1177 |
+
*/
|
| 1178 |
+
public $documentFragment = true;
|
| 1179 |
+
/**
|
| 1180 |
+
* Iterator interface helper
|
| 1181 |
+
* @access private
|
| 1182 |
+
*/
|
| 1183 |
+
protected $elementsInterator = array();
|
| 1184 |
+
/**
|
| 1185 |
+
* Iterator interface helper
|
| 1186 |
+
* @access private
|
| 1187 |
+
*/
|
| 1188 |
+
protected $valid = false;
|
| 1189 |
+
/**
|
| 1190 |
+
* Iterator interface helper
|
| 1191 |
+
* @access private
|
| 1192 |
+
*/
|
| 1193 |
+
protected $current = null;
|
| 1194 |
+
/**
|
| 1195 |
+
* Enter description here...
|
| 1196 |
+
*
|
| 1197 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 1198 |
+
*/
|
| 1199 |
+
public function __construct($documentID) {
|
| 1200 |
+
// if ($documentID instanceof self)
|
| 1201 |
+
// var_dump($documentID->getDocumentID());
|
| 1202 |
+
$id = $documentID instanceof self
|
| 1203 |
+
? $documentID->getDocumentID()
|
| 1204 |
+
: $documentID;
|
| 1205 |
+
// var_dump($id);
|
| 1206 |
+
if (! isset(phpQuery::$documents[$id] )) {
|
| 1207 |
+
// var_dump(phpQuery::$documents);
|
| 1208 |
+
throw new Exception("Document with ID '{$id}' isn't loaded. Use phpQuery::newDocument(\$html) or phpQuery::newDocumentFile(\$file) first.");
|
| 1209 |
+
}
|
| 1210 |
+
$this->documentID = $id;
|
| 1211 |
+
$this->documentWrapper =& phpQuery::$documents[$id];
|
| 1212 |
+
$this->document =& $this->documentWrapper->document;
|
| 1213 |
+
$this->xpath =& $this->documentWrapper->xpath;
|
| 1214 |
+
$this->charset =& $this->documentWrapper->charset;
|
| 1215 |
+
$this->documentFragment =& $this->documentWrapper->isDocumentFragment;
|
| 1216 |
+
// TODO check $this->DOM->documentElement;
|
| 1217 |
+
// $this->root = $this->document->documentElement;
|
| 1218 |
+
$this->root =& $this->documentWrapper->root;
|
| 1219 |
+
// $this->toRoot();
|
| 1220 |
+
$this->elements = array($this->root);
|
| 1221 |
+
}
|
| 1222 |
+
/**
|
| 1223 |
+
*
|
| 1224 |
+
* @access private
|
| 1225 |
+
* @param $attr
|
| 1226 |
+
* @return unknown_type
|
| 1227 |
+
*/
|
| 1228 |
+
public function __get($attr) {
|
| 1229 |
+
switch($attr) {
|
| 1230 |
+
// FIXME doesnt work at all ?
|
| 1231 |
+
case 'length':
|
| 1232 |
+
return $this->size();
|
| 1233 |
+
break;
|
| 1234 |
+
default:
|
| 1235 |
+
return $this->$attr;
|
| 1236 |
+
}
|
| 1237 |
+
}
|
| 1238 |
+
/**
|
| 1239 |
+
* Saves actual object to $var by reference.
|
| 1240 |
+
* Useful when need to break chain.
|
| 1241 |
+
* @param phpQueryObject $var
|
| 1242 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 1243 |
+
*/
|
| 1244 |
+
public function toReference(&$var) {
|
| 1245 |
+
return $var = $this;
|
| 1246 |
+
}
|
| 1247 |
+
public function documentFragment($state = null) {
|
| 1248 |
+
if ($state) {
|
| 1249 |
+
phpQuery::$documents[$this->getDocumentID()]['documentFragment'] = $state;
|
| 1250 |
+
return $this;
|
| 1251 |
+
}
|
| 1252 |
+
return $this->documentFragment;
|
| 1253 |
+
}
|
| 1254 |
+
/**
|
| 1255 |
+
* @access private
|
| 1256 |
+
* @TODO documentWrapper
|
| 1257 |
+
*/
|
| 1258 |
+
protected function isRoot( $node) {
|
| 1259 |
+
// return $node instanceof DOMDOCUMENT || $node->tagName == 'html';
|
| 1260 |
+
return $node instanceof DOMDOCUMENT
|
| 1261 |
+
|| ($node instanceof DOMELEMENT && $node->tagName == 'html')
|
| 1262 |
+
|| $this->root->isSameNode($node);
|
| 1263 |
+
}
|
| 1264 |
+
/**
|
| 1265 |
+
* @access private
|
| 1266 |
+
*/
|
| 1267 |
+
protected function stackIsRoot() {
|
| 1268 |
+
return $this->size() == 1 && $this->isRoot($this->elements[0]);
|
| 1269 |
+
}
|
| 1270 |
+
/**
|
| 1271 |
+
* Enter description here...
|
| 1272 |
+
* NON JQUERY METHOD
|
| 1273 |
+
*
|
| 1274 |
+
* Watch out, it doesn't creates new instance, can be reverted with end().
|
| 1275 |
+
*
|
| 1276 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 1277 |
+
*/
|
| 1278 |
+
public function toRoot() {
|
| 1279 |
+
$this->elements = array($this->root);
|
| 1280 |
+
return $this;
|
| 1281 |
+
// return $this->newInstance(array($this->root));
|
| 1282 |
+
}
|
| 1283 |
+
/**
|
| 1284 |
+
* Saves object's DocumentID to $var by reference.
|
| 1285 |
+
* <code>
|
| 1286 |
+
* $myDocumentId;
|
| 1287 |
+
* phpQuery::newDocument('<div/>')
|
| 1288 |
+
* ->getDocumentIDRef($myDocumentId)
|
| 1289 |
+
* ->find('div')->...
|
| 1290 |
+
* </code>
|
| 1291 |
+
*
|
| 1292 |
+
* @param unknown_type $domId
|
| 1293 |
+
* @see phpQuery::newDocument
|
| 1294 |
+
* @see phpQuery::newDocumentFile
|
| 1295 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 1296 |
+
*/
|
| 1297 |
+
public function getDocumentIDRef(&$documentID) {
|
| 1298 |
+
$documentID = $this->getDocumentID();
|
| 1299 |
+
return $this;
|
| 1300 |
+
}
|
| 1301 |
+
/**
|
| 1302 |
+
* Returns object with stack set to document root.
|
| 1303 |
+
*
|
| 1304 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 1305 |
+
*/
|
| 1306 |
+
public function getDocument() {
|
| 1307 |
+
return phpQuery::getDocument($this->getDocumentID());
|
| 1308 |
+
}
|
| 1309 |
+
/**
|
| 1310 |
+
*
|
| 1311 |
+
* @return DOMDocument
|
| 1312 |
+
*/
|
| 1313 |
+
public function getDOMDocument() {
|
| 1314 |
+
return $this->document;
|
| 1315 |
+
}
|
| 1316 |
+
/**
|
| 1317 |
+
* Get object's Document ID.
|
| 1318 |
+
*
|
| 1319 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 1320 |
+
*/
|
| 1321 |
+
public function getDocumentID() {
|
| 1322 |
+
return $this->documentID;
|
| 1323 |
+
}
|
| 1324 |
+
/**
|
| 1325 |
+
* Unloads whole document from memory.
|
| 1326 |
+
* CAUTION! None further operations will be possible on this document.
|
| 1327 |
+
* All objects refering to it will be useless.
|
| 1328 |
+
*
|
| 1329 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 1330 |
+
*/
|
| 1331 |
+
public function unloadDocument() {
|
| 1332 |
+
phpQuery::unloadDocuments($this->getDocumentID());
|
| 1333 |
+
}
|
| 1334 |
+
public function isHTML() {
|
| 1335 |
+
return $this->documentWrapper->isHTML;
|
| 1336 |
+
}
|
| 1337 |
+
public function isXHTML() {
|
| 1338 |
+
return $this->documentWrapper->isXHTML;
|
| 1339 |
+
}
|
| 1340 |
+
public function isXML() {
|
| 1341 |
+
return $this->documentWrapper->isXML;
|
| 1342 |
+
}
|
| 1343 |
+
/**
|
| 1344 |
+
* Enter description here...
|
| 1345 |
+
*
|
| 1346 |
+
* @link http://docs.jquery.com/Ajax/serialize
|
| 1347 |
+
* @return string
|
| 1348 |
+
*/
|
| 1349 |
+
public function serialize() {
|
| 1350 |
+
return phpQuery::param($this->serializeArray());
|
| 1351 |
+
}
|
| 1352 |
+
/**
|
| 1353 |
+
* Enter description here...
|
| 1354 |
+
*
|
| 1355 |
+
* @link http://docs.jquery.com/Ajax/serializeArray
|
| 1356 |
+
* @return array
|
| 1357 |
+
*/
|
| 1358 |
+
public function serializeArray($submit = null) {
|
| 1359 |
+
$source = $this->filter('form, input, select, textarea')
|
| 1360 |
+
->find('input, select, textarea')
|
| 1361 |
+
->andSelf()
|
| 1362 |
+
->not('form');
|
| 1363 |
+
$return = array();
|
| 1364 |
+
// $source->dumpDie();
|
| 1365 |
+
foreach($source as $input) {
|
| 1366 |
+
$input = phpQuery::pq($input);
|
| 1367 |
+
if ($input->is('[disabled]'))
|
| 1368 |
+
continue;
|
| 1369 |
+
if (!$input->is('[name]'))
|
| 1370 |
+
continue;
|
| 1371 |
+
if ($input->is('[type=checkbox]') && !$input->is('[checked]'))
|
| 1372 |
+
continue;
|
| 1373 |
+
// jquery diff
|
| 1374 |
+
if ($submit && $input->is('[type=submit]')) {
|
| 1375 |
+
if ($submit instanceof DOMELEMENT && ! $input->elements[0]->isSameNode($submit))
|
| 1376 |
+
continue;
|
| 1377 |
+
else if (is_string($submit) && $input->attr('name') != $submit)
|
| 1378 |
+
continue;
|
| 1379 |
+
}
|
| 1380 |
+
$return[] = array(
|
| 1381 |
+
'name' => $input->attr('name'),
|
| 1382 |
+
'value' => $input->val(),
|
| 1383 |
+
);
|
| 1384 |
+
}
|
| 1385 |
+
return $return;
|
| 1386 |
+
}
|
| 1387 |
+
/**
|
| 1388 |
+
* @access private
|
| 1389 |
+
*/
|
| 1390 |
+
protected function debug($in) {
|
| 1391 |
+
if (! phpQuery::$debug )
|
| 1392 |
+
return;
|
| 1393 |
+
print('<pre>');
|
| 1394 |
+
print_r($in);
|
| 1395 |
+
// file debug
|
| 1396 |
+
// file_put_contents(dirname(__FILE__).'/phpQuery.log', print_r($in, true)."\n", FILE_APPEND);
|
| 1397 |
+
// quite handy debug trace
|
| 1398 |
+
// if ( is_array($in))
|
| 1399 |
+
// print_r(array_slice(debug_backtrace(), 3));
|
| 1400 |
+
print("</pre>\n");
|
| 1401 |
+
}
|
| 1402 |
+
/**
|
| 1403 |
+
* @access private
|
| 1404 |
+
*/
|
| 1405 |
+
protected function isRegexp($pattern) {
|
| 1406 |
+
return in_array(
|
| 1407 |
+
$pattern[ mb_strlen($pattern)-1 ],
|
| 1408 |
+
array('^','*','$')
|
| 1409 |
+
);
|
| 1410 |
+
}
|
| 1411 |
+
/**
|
| 1412 |
+
* Determines if $char is really a char.
|
| 1413 |
+
*
|
| 1414 |
+
* @param string $char
|
| 1415 |
+
* @return bool
|
| 1416 |
+
* @todo rewrite me to charcode range ! ;)
|
| 1417 |
+
* @access private
|
| 1418 |
+
*/
|
| 1419 |
+
protected function isChar($char) {
|
| 1420 |
+
return extension_loaded('mbstring') && phpQuery::$mbstringSupport
|
| 1421 |
+
? mb_eregi('\w', $char)
|
| 1422 |
+
: preg_match('@\w@', $char);
|
| 1423 |
+
}
|
| 1424 |
+
/**
|
| 1425 |
+
* @access private
|
| 1426 |
+
*/
|
| 1427 |
+
protected function parseSelector($query) {
|
| 1428 |
+
// clean spaces
|
| 1429 |
+
// TODO include this inside parsing ?
|
| 1430 |
+
$query = trim(
|
| 1431 |
+
preg_replace('@\s+@', ' ',
|
| 1432 |
+
preg_replace('@\s*(>|\\+|~)\s*@', '\\1', $query)
|
| 1433 |
+
)
|
| 1434 |
+
);
|
| 1435 |
+
$queries = array(array());
|
| 1436 |
+
if (! $query)
|
| 1437 |
+
return $queries;
|
| 1438 |
+
$return =& $queries[0];
|
| 1439 |
+
$specialChars = array('>',' ');
|
| 1440 |
+
// $specialCharsMapping = array('/' => '>');
|
| 1441 |
+
$specialCharsMapping = array();
|
| 1442 |
+
$strlen = mb_strlen($query);
|
| 1443 |
+
$classChars = array('.', '-');
|
| 1444 |
+
$pseudoChars = array('-');
|
| 1445 |
+
$tagChars = array('*', '|', '-');
|
| 1446 |
+
// split multibyte string
|
| 1447 |
+
// http://code.google.com/p/phpquery/issues/detail?id=76
|
| 1448 |
+
$_query = array();
|
| 1449 |
+
for ($i=0; $i<$strlen; $i++)
|
| 1450 |
+
$_query[] = mb_substr($query, $i, 1);
|
| 1451 |
+
$query = $_query;
|
| 1452 |
+
// it works, but i dont like it...
|
| 1453 |
+
$i = 0;
|
| 1454 |
+
while( $i < $strlen) {
|
| 1455 |
+
$c = $query[$i];
|
| 1456 |
+
$tmp = '';
|
| 1457 |
+
// TAG
|
| 1458 |
+
if ($this->isChar($c) || in_array($c, $tagChars)) {
|
| 1459 |
+
while(isset($query[$i])
|
| 1460 |
+
&& ($this->isChar($query[$i]) || in_array($query[$i], $tagChars))) {
|
| 1461 |
+
$tmp .= $query[$i];
|
| 1462 |
+
$i++;
|
| 1463 |
+
}
|
| 1464 |
+
$return[] = $tmp;
|
| 1465 |
+
// IDs
|
| 1466 |
+
} else if ( $c == '#') {
|
| 1467 |
+
$i++;
|
| 1468 |
+
while( isset($query[$i]) && ($this->isChar($query[$i]) || $query[$i] == '-')) {
|
| 1469 |
+
$tmp .= $query[$i];
|
| 1470 |
+
$i++;
|
| 1471 |
+
}
|
| 1472 |
+
$return[] = '#'.$tmp;
|
| 1473 |
+
// SPECIAL CHARS
|
| 1474 |
+
} else if (in_array($c, $specialChars)) {
|
| 1475 |
+
$return[] = $c;
|
| 1476 |
+
$i++;
|
| 1477 |
+
// MAPPED SPECIAL MULTICHARS
|
| 1478 |
+
// } else if ( $c.$query[$i+1] == '//') {
|
| 1479 |
+
// $return[] = ' ';
|
| 1480 |
+
// $i = $i+2;
|
| 1481 |
+
// MAPPED SPECIAL CHARS
|
| 1482 |
+
} else if ( isset($specialCharsMapping[$c])) {
|
| 1483 |
+
$return[] = $specialCharsMapping[$c];
|
| 1484 |
+
$i++;
|
| 1485 |
+
// COMMA
|
| 1486 |
+
} else if ( $c == ',') {
|
| 1487 |
+
$queries[] = array();
|
| 1488 |
+
$return =& $queries[ count($queries)-1 ];
|
| 1489 |
+
$i++;
|
| 1490 |
+
while( isset($query[$i]) && $query[$i] == ' ')
|
| 1491 |
+
$i++;
|
| 1492 |
+
// CLASSES
|
| 1493 |
+
} else if ($c == '.') {
|
| 1494 |
+
while( isset($query[$i]) && ($this->isChar($query[$i]) || in_array($query[$i], $classChars))) {
|
| 1495 |
+
$tmp .= $query[$i];
|
| 1496 |
+
$i++;
|
| 1497 |
+
}
|
| 1498 |
+
$return[] = $tmp;
|
| 1499 |
+
// ~ General Sibling Selector
|
| 1500 |
+
} else if ($c == '~') {
|
| 1501 |
+
$spaceAllowed = true;
|
| 1502 |
+
$tmp .= $query[$i++];
|
| 1503 |
+
while( isset($query[$i])
|
| 1504 |
+
&& ($this->isChar($query[$i])
|
| 1505 |
+
|| in_array($query[$i], $classChars)
|
| 1506 |
+
|| $query[$i] == '*'
|
| 1507 |
+
|| ($query[$i] == ' ' && $spaceAllowed)
|
| 1508 |
+
)) {
|
| 1509 |
+
if ($query[$i] != ' ')
|
| 1510 |
+
$spaceAllowed = false;
|
| 1511 |
+
$tmp .= $query[$i];
|
| 1512 |
+
$i++;
|
| 1513 |
+
}
|
| 1514 |
+
$return[] = $tmp;
|
| 1515 |
+
// + Adjacent sibling selectors
|
| 1516 |
+
} else if ($c == '+') {
|
| 1517 |
+
$spaceAllowed = true;
|
| 1518 |
+
$tmp .= $query[$i++];
|
| 1519 |
+
while( isset($query[$i])
|
| 1520 |
+
&& ($this->isChar($query[$i])
|
| 1521 |
+
|| in_array($query[$i], $classChars)
|
| 1522 |
+
|| $query[$i] == '*'
|
| 1523 |
+
|| ($spaceAllowed && $query[$i] == ' ')
|
| 1524 |
+
)) {
|
| 1525 |
+
if ($query[$i] != ' ')
|
| 1526 |
+
$spaceAllowed = false;
|
| 1527 |
+
$tmp .= $query[$i];
|
| 1528 |
+
$i++;
|
| 1529 |
+
}
|
| 1530 |
+
$return[] = $tmp;
|
| 1531 |
+
// ATTRS
|
| 1532 |
+
} else if ($c == '[') {
|
| 1533 |
+
$stack = 1;
|
| 1534 |
+
$tmp .= $c;
|
| 1535 |
+
while( isset($query[++$i])) {
|
| 1536 |
+
$tmp .= $query[$i];
|
| 1537 |
+
if ( $query[$i] == '[') {
|
| 1538 |
+
$stack++;
|
| 1539 |
+
} else if ( $query[$i] == ']') {
|
| 1540 |
+
$stack--;
|
| 1541 |
+
if (! $stack )
|
| 1542 |
+
break;
|
| 1543 |
+
}
|
| 1544 |
+
}
|
| 1545 |
+
$return[] = $tmp;
|
| 1546 |
+
$i++;
|
| 1547 |
+
// PSEUDO CLASSES
|
| 1548 |
+
} else if ($c == ':') {
|
| 1549 |
+
$stack = 1;
|
| 1550 |
+
$tmp .= $query[$i++];
|
| 1551 |
+
while( isset($query[$i]) && ($this->isChar($query[$i]) || in_array($query[$i], $pseudoChars))) {
|
| 1552 |
+
$tmp .= $query[$i];
|
| 1553 |
+
$i++;
|
| 1554 |
+
}
|
| 1555 |
+
// with arguments ?
|
| 1556 |
+
if ( isset($query[$i]) && $query[$i] == '(') {
|
| 1557 |
+
$tmp .= $query[$i];
|
| 1558 |
+
$stack = 1;
|
| 1559 |
+
while( isset($query[++$i])) {
|
| 1560 |
+
$tmp .= $query[$i];
|
| 1561 |
+
if ( $query[$i] == '(') {
|
| 1562 |
+
$stack++;
|
| 1563 |
+
} else if ( $query[$i] == ')') {
|
| 1564 |
+
$stack--;
|
| 1565 |
+
if (! $stack )
|
| 1566 |
+
break;
|
| 1567 |
+
}
|
| 1568 |
+
}
|
| 1569 |
+
$return[] = $tmp;
|
| 1570 |
+
$i++;
|
| 1571 |
+
} else {
|
| 1572 |
+
$return[] = $tmp;
|
| 1573 |
+
}
|
| 1574 |
+
} else {
|
| 1575 |
+
$i++;
|
| 1576 |
+
}
|
| 1577 |
+
}
|
| 1578 |
+
foreach($queries as $k => $q) {
|
| 1579 |
+
if (isset($q[0])) {
|
| 1580 |
+
if (isset($q[0][0]) && $q[0][0] == ':')
|
| 1581 |
+
array_unshift($queries[$k], '*');
|
| 1582 |
+
if ($q[0] != '>')
|
| 1583 |
+
array_unshift($queries[$k], ' ');
|
| 1584 |
+
}
|
| 1585 |
+
}
|
| 1586 |
+
return $queries;
|
| 1587 |
+
}
|
| 1588 |
+
/**
|
| 1589 |
+
* Return matched DOM nodes.
|
| 1590 |
+
*
|
| 1591 |
+
* @param int $index
|
| 1592 |
+
* @return array|DOMElement Single DOMElement or array of DOMElement.
|
| 1593 |
+
*/
|
| 1594 |
+
public function get($index = null, $callback1 = null, $callback2 = null, $callback3 = null) {
|
| 1595 |
+
$return = isset($index)
|
| 1596 |
+
? (isset($this->elements[$index]) ? $this->elements[$index] : null)
|
| 1597 |
+
: $this->elements;
|
| 1598 |
+
// pass thou callbacks
|
| 1599 |
+
$args = func_get_args();
|
| 1600 |
+
$args = array_slice($args, 1);
|
| 1601 |
+
foreach($args as $callback) {
|
| 1602 |
+
if (is_array($return))
|
| 1603 |
+
foreach($return as $k => $v)
|
| 1604 |
+
$return[$k] = phpQuery::callbackRun($callback, array($v));
|
| 1605 |
+
else
|
| 1606 |
+
$return = phpQuery::callbackRun($callback, array($return));
|
| 1607 |
+
}
|
| 1608 |
+
return $return;
|
| 1609 |
+
}
|
| 1610 |
+
/**
|
| 1611 |
+
* Return matched DOM nodes.
|
| 1612 |
+
* jQuery difference.
|
| 1613 |
+
*
|
| 1614 |
+
* @param int $index
|
| 1615 |
+
* @return array|string Returns string if $index != null
|
| 1616 |
+
* @todo implement callbacks
|
| 1617 |
+
* @todo return only arrays ?
|
| 1618 |
+
* @todo maybe other name...
|
| 1619 |
+
*/
|
| 1620 |
+
public function getString($index = null, $callback1 = null, $callback2 = null, $callback3 = null) {
|
| 1621 |
+
if ($index)
|
| 1622 |
+
$return = $this->eq($index)->text();
|
| 1623 |
+
else {
|
| 1624 |
+
$return = array();
|
| 1625 |
+
for($i = 0; $i < $this->size(); $i++) {
|
| 1626 |
+
$return[] = $this->eq($i)->text();
|
| 1627 |
+
}
|
| 1628 |
+
}
|
| 1629 |
+
// pass thou callbacks
|
| 1630 |
+
$args = func_get_args();
|
| 1631 |
+
$args = array_slice($args, 1);
|
| 1632 |
+
foreach($args as $callback) {
|
| 1633 |
+
$return = phpQuery::callbackRun($callback, array($return));
|
| 1634 |
+
}
|
| 1635 |
+
return $return;
|
| 1636 |
+
}
|
| 1637 |
+
/**
|
| 1638 |
+
* Return matched DOM nodes.
|
| 1639 |
+
* jQuery difference.
|
| 1640 |
+
*
|
| 1641 |
+
* @param int $index
|
| 1642 |
+
* @return array|string Returns string if $index != null
|
| 1643 |
+
* @todo implement callbacks
|
| 1644 |
+
* @todo return only arrays ?
|
| 1645 |
+
* @todo maybe other name...
|
| 1646 |
+
*/
|
| 1647 |
+
public function getStrings($index = null, $callback1 = null, $callback2 = null, $callback3 = null) {
|
| 1648 |
+
if ($index)
|
| 1649 |
+
$return = $this->eq($index)->text();
|
| 1650 |
+
else {
|
| 1651 |
+
$return = array();
|
| 1652 |
+
for($i = 0; $i < $this->size(); $i++) {
|
| 1653 |
+
$return[] = $this->eq($i)->text();
|
| 1654 |
+
}
|
| 1655 |
+
// pass thou callbacks
|
| 1656 |
+
$args = func_get_args();
|
| 1657 |
+
$args = array_slice($args, 1);
|
| 1658 |
+
}
|
| 1659 |
+
foreach($args as $callback) {
|
| 1660 |
+
if (is_array($return))
|
| 1661 |
+
foreach($return as $k => $v)
|
| 1662 |
+
$return[$k] = phpQuery::callbackRun($callback, array($v));
|
| 1663 |
+
else
|
| 1664 |
+
$return = phpQuery::callbackRun($callback, array($return));
|
| 1665 |
+
}
|
| 1666 |
+
return $return;
|
| 1667 |
+
}
|
| 1668 |
+
/**
|
| 1669 |
+
* Returns new instance of actual class.
|
| 1670 |
+
*
|
| 1671 |
+
* @param array $newStack Optional. Will replace old stack with new and move old one to history.c
|
| 1672 |
+
*/
|
| 1673 |
+
public function newInstance($newStack = null) {
|
| 1674 |
+
$class = get_class($this);
|
| 1675 |
+
// support inheritance by passing old object to overloaded constructor
|
| 1676 |
+
$new = $class != 'phpQuery'
|
| 1677 |
+
? new $class($this, $this->getDocumentID())
|
| 1678 |
+
: new phpQueryObject($this->getDocumentID());
|
| 1679 |
+
$new->previous = $this;
|
| 1680 |
+
if (is_null($newStack)) {
|
| 1681 |
+
$new->elements = $this->elements;
|
| 1682 |
+
if ($this->elementsBackup)
|
| 1683 |
+
$this->elements = $this->elementsBackup;
|
| 1684 |
+
} else if (is_string($newStack)) {
|
| 1685 |
+
$new->elements = phpQuery::pq($newStack, $this->getDocumentID())->stack();
|
| 1686 |
+
} else {
|
| 1687 |
+
$new->elements = $newStack;
|
| 1688 |
+
}
|
| 1689 |
+
return $new;
|
| 1690 |
+
}
|
| 1691 |
+
/**
|
| 1692 |
+
* Enter description here...
|
| 1693 |
+
*
|
| 1694 |
+
* In the future, when PHP will support XLS 2.0, then we would do that this way:
|
| 1695 |
+
* contains(tokenize(@class, '\s'), "something")
|
| 1696 |
+
* @param unknown_type $class
|
| 1697 |
+
* @param unknown_type $node
|
| 1698 |
+
* @return boolean
|
| 1699 |
+
* @access private
|
| 1700 |
+
*/
|
| 1701 |
+
protected function matchClasses($class, $node) {
|
| 1702 |
+
// multi-class
|
| 1703 |
+
if ( mb_strpos($class, '.', 1)) {
|
| 1704 |
+
$classes = explode('.', substr($class, 1));
|
| 1705 |
+
$classesCount = count( $classes );
|
| 1706 |
+
$nodeClasses = explode(' ', $node->getAttribute('class') );
|
| 1707 |
+
$nodeClassesCount = count( $nodeClasses );
|
| 1708 |
+
if ( $classesCount > $nodeClassesCount )
|
| 1709 |
+
return false;
|
| 1710 |
+
$diff = count(
|
| 1711 |
+
array_diff(
|
| 1712 |
+
$classes,
|
| 1713 |
+
$nodeClasses
|
| 1714 |
+
)
|
| 1715 |
+
);
|
| 1716 |
+
if (! $diff )
|
| 1717 |
+
return true;
|
| 1718 |
+
// single-class
|
| 1719 |
+
} else {
|
| 1720 |
+
return in_array(
|
| 1721 |
+
// strip leading dot from class name
|
| 1722 |
+
substr($class, 1),
|
| 1723 |
+
// get classes for element as array
|
| 1724 |
+
explode(' ', $node->getAttribute('class') )
|
| 1725 |
+
);
|
| 1726 |
+
}
|
| 1727 |
+
}
|
| 1728 |
+
/**
|
| 1729 |
+
* @access private
|
| 1730 |
+
*/
|
| 1731 |
+
protected function runQuery($XQuery, $selector = null, $compare = null) {
|
| 1732 |
+
if ($compare && ! method_exists($this, $compare))
|
| 1733 |
+
return false;
|
| 1734 |
+
$stack = array();
|
| 1735 |
+
if (! $this->elements)
|
| 1736 |
+
$this->debug('Stack empty, skipping...');
|
| 1737 |
+
// var_dump($this->elements[0]->nodeType);
|
| 1738 |
+
// element, document
|
| 1739 |
+
foreach($this->stack(array(1, 9, 13)) as $k => $stackNode) {
|
| 1740 |
+
$detachAfter = false;
|
| 1741 |
+
// to work on detached nodes we need temporary place them somewhere
|
| 1742 |
+
// thats because context xpath queries sucks ;]
|
| 1743 |
+
$testNode = $stackNode;
|
| 1744 |
+
while ($testNode) {
|
| 1745 |
+
if (! $testNode->parentNode && ! $this->isRoot($testNode)) {
|
| 1746 |
+
$this->root->appendChild($testNode);
|
| 1747 |
+
$detachAfter = $testNode;
|
| 1748 |
+
break;
|
| 1749 |
+
}
|
| 1750 |
+
$testNode = isset($testNode->parentNode)
|
| 1751 |
+
? $testNode->parentNode
|
| 1752 |
+
: null;
|
| 1753 |
+
}
|
| 1754 |
+
// XXX tmp ?
|
| 1755 |
+
$xpath = $this->documentWrapper->isXHTML
|
| 1756 |
+
? $this->getNodeXpath($stackNode, 'html')
|
| 1757 |
+
: $this->getNodeXpath($stackNode);
|
| 1758 |
+
// FIXME pseudoclasses-only query, support XML
|
| 1759 |
+
$query = $XQuery == '//' && $xpath == '/html[1]'
|
| 1760 |
+
? '//*'
|
| 1761 |
+
: $xpath.$XQuery;
|
| 1762 |
+
$this->debug("XPATH: {$query}");
|
| 1763 |
+
// run query, get elements
|
| 1764 |
+
$nodes = $this->xpath->query($query);
|
| 1765 |
+
$this->debug("QUERY FETCHED");
|
| 1766 |
+
if (! $nodes->length )
|
| 1767 |
+
$this->debug('Nothing found');
|
| 1768 |
+
$debug = array();
|
| 1769 |
+
foreach($nodes as $node) {
|
| 1770 |
+
$matched = false;
|
| 1771 |
+
if ( $compare) {
|
| 1772 |
+
phpQuery::$debug ?
|
| 1773 |
+
$this->debug("Found: ".$this->whois( $node ).", comparing with {$compare}()")
|
| 1774 |
+
: null;
|
| 1775 |
+
$phpQueryDebug = phpQuery::$debug;
|
| 1776 |
+
phpQuery::$debug = false;
|
| 1777 |
+
// TODO ??? use phpQuery::callbackRun()
|
| 1778 |
+
if (call_user_func_array(array($this, $compare), array($selector, $node)))
|
| 1779 |
+
$matched = true;
|
| 1780 |
+
phpQuery::$debug = $phpQueryDebug;
|
| 1781 |
+
} else {
|
| 1782 |
+
$matched = true;
|
| 1783 |
+
}
|
| 1784 |
+
if ( $matched) {
|
| 1785 |
+
if (phpQuery::$debug)
|
| 1786 |
+
$debug[] = $this->whois( $node );
|
| 1787 |
+
$stack[] = $node;
|
| 1788 |
+
}
|
| 1789 |
+
}
|
| 1790 |
+
if (phpQuery::$debug) {
|
| 1791 |
+
$this->debug("Matched ".count($debug).": ".implode(', ', $debug));
|
| 1792 |
+
}
|
| 1793 |
+
if ($detachAfter)
|
| 1794 |
+
$this->root->removeChild($detachAfter);
|
| 1795 |
+
}
|
| 1796 |
+
$this->elements = $stack;
|
| 1797 |
+
}
|
| 1798 |
+
/**
|
| 1799 |
+
* Enter description here...
|
| 1800 |
+
*
|
| 1801 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 1802 |
+
*/
|
| 1803 |
+
public function find($selectors, $context = null, $noHistory = false) {
|
| 1804 |
+
if (!$noHistory)
|
| 1805 |
+
// backup last stack /for end()/
|
| 1806 |
+
$this->elementsBackup = $this->elements;
|
| 1807 |
+
// allow to define context
|
| 1808 |
+
// TODO combine code below with phpQuery::pq() context guessing code
|
| 1809 |
+
// as generic function
|
| 1810 |
+
if ($context) {
|
| 1811 |
+
if (! is_array($context) && $context instanceof DOMELEMENT)
|
| 1812 |
+
$this->elements = array($context);
|
| 1813 |
+
else if (is_array($context)) {
|
| 1814 |
+
$this->elements = array();
|
| 1815 |
+
foreach ($context as $c)
|
| 1816 |
+
if ($c instanceof DOMELEMENT)
|
| 1817 |
+
$this->elements[] = $c;
|
| 1818 |
+
} else if ( $context instanceof self )
|
| 1819 |
+
$this->elements = $context->elements;
|
| 1820 |
+
}
|
| 1821 |
+
$queries = $this->parseSelector($selectors);
|
| 1822 |
+
$this->debug(array('FIND', $selectors, $queries));
|
| 1823 |
+
$XQuery = '';
|
| 1824 |
+
// remember stack state because of multi-queries
|
| 1825 |
+
$oldStack = $this->elements;
|
| 1826 |
+
// here we will be keeping found elements
|
| 1827 |
+
$stack = array();
|
| 1828 |
+
foreach($queries as $selector) {
|
| 1829 |
+
$this->elements = $oldStack;
|
| 1830 |
+
$delimiterBefore = false;
|
| 1831 |
+
foreach($selector as $s) {
|
| 1832 |
+
// TAG
|
| 1833 |
+
$isTag = extension_loaded('mbstring') && phpQuery::$mbstringSupport
|
| 1834 |
+
? mb_ereg_match('^[\w|\||-]+$', $s) || $s == '*'
|
| 1835 |
+
: preg_match('@^[\w|\||-]+$@', $s) || $s == '*';
|
| 1836 |
+
if ($isTag) {
|
| 1837 |
+
if ($this->isXML()) {
|
| 1838 |
+
// namespace support
|
| 1839 |
+
if (mb_strpos($s, '|') !== false) {
|
| 1840 |
+
$ns = $tag = null;
|
| 1841 |
+
list($ns, $tag) = explode('|', $s);
|
| 1842 |
+
$XQuery .= "$ns:$tag";
|
| 1843 |
+
} else if ($s == '*') {
|
| 1844 |
+
$XQuery .= "*";
|
| 1845 |
+
} else {
|
| 1846 |
+
$XQuery .= "*[local-name()='$s']";
|
| 1847 |
+
}
|
| 1848 |
+
} else {
|
| 1849 |
+
$XQuery .= $s;
|
| 1850 |
+
}
|
| 1851 |
+
// ID
|
| 1852 |
+
} else if ($s[0] == '#') {
|
| 1853 |
+
if ($delimiterBefore)
|
| 1854 |
+
$XQuery .= '*';
|
| 1855 |
+
$XQuery .= "[@id='".substr($s, 1)."']";
|
| 1856 |
+
// ATTRIBUTES
|
| 1857 |
+
} else if ($s[0] == '[') {
|
| 1858 |
+
if ($delimiterBefore)
|
| 1859 |
+
$XQuery .= '*';
|
| 1860 |
+
// strip side brackets
|
| 1861 |
+
$attr = trim($s, '][');
|
| 1862 |
+
$execute = false;
|
| 1863 |
+
// attr with specifed value
|
| 1864 |
+
if (mb_strpos($s, '=')) {
|
| 1865 |
+
$value = null;
|
| 1866 |
+
list($attr, $value) = explode('=', $attr);
|
| 1867 |
+
$value = trim($value, "'\"");
|
| 1868 |
+
if ($this->isRegexp($attr)) {
|
| 1869 |
+
// cut regexp character
|
| 1870 |
+
$attr = substr($attr, 0, -1);
|
| 1871 |
+
$execute = true;
|
| 1872 |
+
$XQuery .= "[@{$attr}]";
|
| 1873 |
+
} else {
|
| 1874 |
+
$XQuery .= "[@{$attr}='{$value}']";
|
| 1875 |
+
}
|
| 1876 |
+
// attr without specified value
|
| 1877 |
+
} else {
|
| 1878 |
+
$XQuery .= "[@{$attr}]";
|
| 1879 |
+
}
|
| 1880 |
+
if ($execute) {
|
| 1881 |
+
$this->runQuery($XQuery, $s, 'is');
|
| 1882 |
+
$XQuery = '';
|
| 1883 |
+
if (! $this->length())
|
| 1884 |
+
break;
|
| 1885 |
+
}
|
| 1886 |
+
// CLASSES
|
| 1887 |
+
} else if ($s[0] == '.') {
|
| 1888 |
+
// TODO use return $this->find("./self::*[contains(concat(\" \",@class,\" \"), \" $class \")]");
|
| 1889 |
+
// thx wizDom ;)
|
| 1890 |
+
if ($delimiterBefore)
|
| 1891 |
+
$XQuery .= '*';
|
| 1892 |
+
$XQuery .= '[@class]';
|
| 1893 |
+
$this->runQuery($XQuery, $s, 'matchClasses');
|
| 1894 |
+
$XQuery = '';
|
| 1895 |
+
if (! $this->length() )
|
| 1896 |
+
break;
|
| 1897 |
+
// ~ General Sibling Selector
|
| 1898 |
+
} else if ($s[0] == '~') {
|
| 1899 |
+
$this->runQuery($XQuery);
|
| 1900 |
+
$XQuery = '';
|
| 1901 |
+
$this->elements = $this
|
| 1902 |
+
->siblings(
|
| 1903 |
+
substr($s, 1)
|
| 1904 |
+
)->elements;
|
| 1905 |
+
if (! $this->length() )
|
| 1906 |
+
break;
|
| 1907 |
+
// + Adjacent sibling selectors
|
| 1908 |
+
} else if ($s[0] == '+') {
|
| 1909 |
+
// TODO /following-sibling::
|
| 1910 |
+
$this->runQuery($XQuery);
|
| 1911 |
+
$XQuery = '';
|
| 1912 |
+
$subSelector = substr($s, 1);
|
| 1913 |
+
$subElements = $this->elements;
|
| 1914 |
+
$this->elements = array();
|
| 1915 |
+
foreach($subElements as $node) {
|
| 1916 |
+
// search first DOMElement sibling
|
| 1917 |
+
$test = $node->nextSibling;
|
| 1918 |
+
while($test && ! ($test instanceof DOMELEMENT))
|
| 1919 |
+
$test = $test->nextSibling;
|
| 1920 |
+
if ($test && $this->is($subSelector, $test))
|
| 1921 |
+
$this->elements[] = $test;
|
| 1922 |
+
}
|
| 1923 |
+
if (! $this->length() )
|
| 1924 |
+
break;
|
| 1925 |
+
// PSEUDO CLASSES
|
| 1926 |
+
} else if ($s[0] == ':') {
|
| 1927 |
+
// TODO optimization for :first :last
|
| 1928 |
+
if ($XQuery) {
|
| 1929 |
+
$this->runQuery($XQuery);
|
| 1930 |
+
$XQuery = '';
|
| 1931 |
+
}
|
| 1932 |
+
if (! $this->length())
|
| 1933 |
+
break;
|
| 1934 |
+
$this->pseudoClasses($s);
|
| 1935 |
+
if (! $this->length())
|
| 1936 |
+
break;
|
| 1937 |
+
// DIRECT DESCENDANDS
|
| 1938 |
+
} else if ($s == '>') {
|
| 1939 |
+
$XQuery .= '/';
|
| 1940 |
+
$delimiterBefore = 2;
|
| 1941 |
+
// ALL DESCENDANDS
|
| 1942 |
+
} else if ($s == ' ') {
|
| 1943 |
+
$XQuery .= '//';
|
| 1944 |
+
$delimiterBefore = 2;
|
| 1945 |
+
// ERRORS
|
| 1946 |
+
} else {
|
| 1947 |
+
phpQuery::debug("Unrecognized token '$s'");
|
| 1948 |
+
}
|
| 1949 |
+
$delimiterBefore = $delimiterBefore === 2;
|
| 1950 |
+
}
|
| 1951 |
+
// run query if any
|
| 1952 |
+
if ($XQuery && $XQuery != '//') {
|
| 1953 |
+
$this->runQuery($XQuery);
|
| 1954 |
+
$XQuery = '';
|
| 1955 |
+
}
|
| 1956 |
+
foreach($this->elements as $node)
|
| 1957 |
+
if (! $this->elementsContainsNode($node, $stack))
|
| 1958 |
+
$stack[] = $node;
|
| 1959 |
+
}
|
| 1960 |
+
$this->elements = $stack;
|
| 1961 |
+
return $this->newInstance();
|
| 1962 |
+
}
|
| 1963 |
+
/**
|
| 1964 |
+
* @todo create API for classes with pseudoselectors
|
| 1965 |
+
* @access private
|
| 1966 |
+
*/
|
| 1967 |
+
protected function pseudoClasses($class) {
|
| 1968 |
+
// TODO clean args parsing ?
|
| 1969 |
+
$class = ltrim($class, ':');
|
| 1970 |
+
$haveArgs = mb_strpos($class, '(');
|
| 1971 |
+
if ($haveArgs !== false) {
|
| 1972 |
+
$args = substr($class, $haveArgs+1, -1);
|
| 1973 |
+
$class = substr($class, 0, $haveArgs);
|
| 1974 |
+
}
|
| 1975 |
+
switch($class) {
|
| 1976 |
+
case 'even':
|
| 1977 |
+
case 'odd':
|
| 1978 |
+
$stack = array();
|
| 1979 |
+
foreach($this->elements as $i => $node) {
|
| 1980 |
+
if ($class == 'even' && ($i%2) == 0)
|
| 1981 |
+
$stack[] = $node;
|
| 1982 |
+
else if ( $class == 'odd' && $i % 2 )
|
| 1983 |
+
$stack[] = $node;
|
| 1984 |
+
}
|
| 1985 |
+
$this->elements = $stack;
|
| 1986 |
+
break;
|
| 1987 |
+
case 'eq':
|
| 1988 |
+
$k = intval($args);
|
| 1989 |
+
$this->elements = isset( $this->elements[$k] )
|
| 1990 |
+
? array( $this->elements[$k] )
|
| 1991 |
+
: array();
|
| 1992 |
+
break;
|
| 1993 |
+
case 'gt':
|
| 1994 |
+
$this->elements = array_slice($this->elements, $args+1);
|
| 1995 |
+
break;
|
| 1996 |
+
case 'lt':
|
| 1997 |
+
$this->elements = array_slice($this->elements, 0, $args+1);
|
| 1998 |
+
break;
|
| 1999 |
+
case 'first':
|
| 2000 |
+
if (isset($this->elements[0]))
|
| 2001 |
+
$this->elements = array($this->elements[0]);
|
| 2002 |
+
break;
|
| 2003 |
+
case 'last':
|
| 2004 |
+
if ($this->elements)
|
| 2005 |
+
$this->elements = array($this->elements[count($this->elements)-1]);
|
| 2006 |
+
break;
|
| 2007 |
+
/*case 'parent':
|
| 2008 |
+
$stack = array();
|
| 2009 |
+
foreach($this->elements as $node) {
|
| 2010 |
+
if ( $node->childNodes->length )
|
| 2011 |
+
$stack[] = $node;
|
| 2012 |
+
}
|
| 2013 |
+
$this->elements = $stack;
|
| 2014 |
+
break;*/
|
| 2015 |
+
case 'contains':
|
| 2016 |
+
$text = trim($args, "\"'");
|
| 2017 |
+
$stack = array();
|
| 2018 |
+
foreach($this->elements as $node) {
|
| 2019 |
+
if (mb_stripos($node->textContent, $text) === false)
|
| 2020 |
+
continue;
|
| 2021 |
+
$stack[] = $node;
|
| 2022 |
+
}
|
| 2023 |
+
$this->elements = $stack;
|
| 2024 |
+
break;
|
| 2025 |
+
case 'not':
|
| 2026 |
+
$selector = self::unQuote($args);
|
| 2027 |
+
$this->elements = $this->not($selector)->stack();
|
| 2028 |
+
break;
|
| 2029 |
+
case 'slice':
|
| 2030 |
+
// TODO jQuery difference ?
|
| 2031 |
+
$args = explode(',',
|
| 2032 |
+
str_replace(', ', ',', trim($args, "\"'"))
|
| 2033 |
+
);
|
| 2034 |
+
$start = $args[0];
|
| 2035 |
+
$end = isset($args[1])
|
| 2036 |
+
? $args[1]
|
| 2037 |
+
: null;
|
| 2038 |
+
if ($end > 0)
|
| 2039 |
+
$end = $end-$start;
|
| 2040 |
+
$this->elements = array_slice($this->elements, $start, $end);
|
| 2041 |
+
break;
|
| 2042 |
+
case 'has':
|
| 2043 |
+
$selector = trim($args, "\"'");
|
| 2044 |
+
$stack = array();
|
| 2045 |
+
foreach($this->stack(1) as $el) {
|
| 2046 |
+
if ($this->find($selector, $el, true)->length)
|
| 2047 |
+
$stack[] = $el;
|
| 2048 |
+
}
|
| 2049 |
+
$this->elements = $stack;
|
| 2050 |
+
break;
|
| 2051 |
+
case 'submit':
|
| 2052 |
+
case 'reset':
|
| 2053 |
+
$this->elements = phpQuery::merge(
|
| 2054 |
+
$this->map(array($this, 'is'),
|
| 2055 |
+
"input[type=$class]", new CallbackParam()
|
| 2056 |
+
),
|
| 2057 |
+
$this->map(array($this, 'is'),
|
| 2058 |
+
"button[type=$class]", new CallbackParam()
|
| 2059 |
+
)
|
| 2060 |
+
);
|
| 2061 |
+
break;
|
| 2062 |
+
// $stack = array();
|
| 2063 |
+
// foreach($this->elements as $node)
|
| 2064 |
+
// if ($node->is('input[type=submit]') || $node->is('button[type=submit]'))
|
| 2065 |
+
// $stack[] = $el;
|
| 2066 |
+
// $this->elements = $stack;
|
| 2067 |
+
case 'input':
|
| 2068 |
+
$this->elements = $this->map(
|
| 2069 |
+
array($this, 'is'),
|
| 2070 |
+
'input', new CallbackParam()
|
| 2071 |
+
)->elements;
|
| 2072 |
+
break;
|
| 2073 |
+
case 'password':
|
| 2074 |
+
case 'checkbox':
|
| 2075 |
+
case 'radio':
|
| 2076 |
+
case 'hidden':
|
| 2077 |
+
case 'image':
|
| 2078 |
+
case 'file':
|
| 2079 |
+
$this->elements = $this->map(
|
| 2080 |
+
array($this, 'is'),
|
| 2081 |
+
"input[type=$class]", new CallbackParam()
|
| 2082 |
+
)->elements;
|
| 2083 |
+
break;
|
| 2084 |
+
case 'parent':
|
| 2085 |
+
$this->elements = $this->map(
|
| 2086 |
+
create_function('$node', '
|
| 2087 |
+
return $node instanceof DOMELEMENT && $node->childNodes->length
|
| 2088 |
+
? $node : null;')
|
| 2089 |
+
)->elements;
|
| 2090 |
+
break;
|
| 2091 |
+
case 'empty':
|
| 2092 |
+
$this->elements = $this->map(
|
| 2093 |
+
create_function('$node', '
|
| 2094 |
+
return $node instanceof DOMELEMENT && $node->childNodes->length
|
| 2095 |
+
? null : $node;')
|
| 2096 |
+
)->elements;
|
| 2097 |
+
break;
|
| 2098 |
+
case 'disabled':
|
| 2099 |
+
case 'selected':
|
| 2100 |
+
case 'checked':
|
| 2101 |
+
$this->elements = $this->map(
|
| 2102 |
+
array($this, 'is'),
|
| 2103 |
+
"[$class]", new CallbackParam()
|
| 2104 |
+
)->elements;
|
| 2105 |
+
break;
|
| 2106 |
+
case 'enabled':
|
| 2107 |
+
$this->elements = $this->map(
|
| 2108 |
+
create_function('$node', '
|
| 2109 |
+
return pq($node)->not(":disabled") ? $node : null;')
|
| 2110 |
+
)->elements;
|
| 2111 |
+
break;
|
| 2112 |
+
case 'header':
|
| 2113 |
+
$this->elements = $this->map(
|
| 2114 |
+
create_function('$node',
|
| 2115 |
+
'$isHeader = isset($node->tagName) && in_array($node->tagName, array(
|
| 2116 |
+
"h1", "h2", "h3", "h4", "h5", "h6", "h7"
|
| 2117 |
+
));
|
| 2118 |
+
return $isHeader
|
| 2119 |
+
? $node
|
| 2120 |
+
: null;')
|
| 2121 |
+
)->elements;
|
| 2122 |
+
// $this->elements = $this->map(
|
| 2123 |
+
// create_function('$node', '$node = pq($node);
|
| 2124 |
+
// return $node->is("h1")
|
| 2125 |
+
// || $node->is("h2")
|
| 2126 |
+
// || $node->is("h3")
|
| 2127 |
+
// || $node->is("h4")
|
| 2128 |
+
// || $node->is("h5")
|
| 2129 |
+
// || $node->is("h6")
|
| 2130 |
+
// || $node->is("h7")
|
| 2131 |
+
// ? $node
|
| 2132 |
+
// : null;')
|
| 2133 |
+
// )->elements;
|
| 2134 |
+
break;
|
| 2135 |
+
case 'only-child':
|
| 2136 |
+
$this->elements = $this->map(
|
| 2137 |
+
create_function('$node',
|
| 2138 |
+
'return pq($node)->siblings()->size() == 0 ? $node : null;')
|
| 2139 |
+
)->elements;
|
| 2140 |
+
break;
|
| 2141 |
+
case 'first-child':
|
| 2142 |
+
$this->elements = $this->map(
|
| 2143 |
+
create_function('$node', 'return pq($node)->prevAll()->size() == 0 ? $node : null;')
|
| 2144 |
+
)->elements;
|
| 2145 |
+
break;
|
| 2146 |
+
case 'last-child':
|
| 2147 |
+
$this->elements = $this->map(
|
| 2148 |
+
create_function('$node', 'return pq($node)->nextAll()->size() == 0 ? $node : null;')
|
| 2149 |
+
)->elements;
|
| 2150 |
+
break;
|
| 2151 |
+
case 'nth-child':
|
| 2152 |
+
$param = trim($args, "\"'");
|
| 2153 |
+
if (! $param)
|
| 2154 |
+
break;
|
| 2155 |
+
// nth-child(n+b) to nth-child(1n+b)
|
| 2156 |
+
if ($param{0} == 'n')
|
| 2157 |
+
$param = '1'.$param;
|
| 2158 |
+
// :nth-child(index/even/odd/equation)
|
| 2159 |
+
if ($param == 'even' || $param == 'odd')
|
| 2160 |
+
$mapped = $this->map(
|
| 2161 |
+
create_function('$node, $param',
|
| 2162 |
+
'$index = pq($node)->prevAll()->size()+1;
|
| 2163 |
+
if ($param == "even" && ($index%2) == 0)
|
| 2164 |
+
return $node;
|
| 2165 |
+
else if ($param == "odd" && $index%2 == 1)
|
| 2166 |
+
return $node;
|
| 2167 |
+
else
|
| 2168 |
+
return null;'),
|
| 2169 |
+
new CallbackParam(), $param
|
| 2170 |
+
);
|
| 2171 |
+
else if (mb_strlen($param) > 1 && $param{1} == 'n')
|
| 2172 |
+
// an+b
|
| 2173 |
+
$mapped = $this->map(
|
| 2174 |
+
create_function('$node, $param',
|
| 2175 |
+
'$prevs = pq($node)->prevAll()->size();
|
| 2176 |
+
$index = 1+$prevs;
|
| 2177 |
+
$b = mb_strlen($param) > 3
|
| 2178 |
+
? $param{3}
|
| 2179 |
+
: 0;
|
| 2180 |
+
$a = $param{0};
|
| 2181 |
+
if ($b && $param{2} == "-")
|
| 2182 |
+
$b = -$b;
|
| 2183 |
+
if ($a > 0) {
|
| 2184 |
+
return ($index-$b)%$a == 0
|
| 2185 |
+
? $node
|
| 2186 |
+
: null;
|
| 2187 |
+
phpQuery::debug($a."*".floor($index/$a)."+$b-1 == ".($a*floor($index/$a)+$b-1)." ?= $prevs");
|
| 2188 |
+
return $a*floor($index/$a)+$b-1 == $prevs
|
| 2189 |
+
? $node
|
| 2190 |
+
: null;
|
| 2191 |
+
} else if ($a == 0)
|
| 2192 |
+
return $index == $b
|
| 2193 |
+
? $node
|
| 2194 |
+
: null;
|
| 2195 |
+
else
|
| 2196 |
+
// negative value
|
| 2197 |
+
return $index <= $b
|
| 2198 |
+
? $node
|
| 2199 |
+
: null;
|
| 2200 |
+
// if (! $b)
|
| 2201 |
+
// return $index%$a == 0
|
| 2202 |
+
// ? $node
|
| 2203 |
+
// : null;
|
| 2204 |
+
// else
|
| 2205 |
+
// return ($index-$b)%$a == 0
|
| 2206 |
+
// ? $node
|
| 2207 |
+
// : null;
|
| 2208 |
+
'),
|
| 2209 |
+
new CallbackParam(), $param
|
| 2210 |
+
);
|
| 2211 |
+
else
|
| 2212 |
+
// index
|
| 2213 |
+
$mapped = $this->map(
|
| 2214 |
+
create_function('$node, $index',
|
| 2215 |
+
'$prevs = pq($node)->prevAll()->size();
|
| 2216 |
+
if ($prevs && $prevs == $index-1)
|
| 2217 |
+
return $node;
|
| 2218 |
+
else if (! $prevs && $index == 1)
|
| 2219 |
+
return $node;
|
| 2220 |
+
else
|
| 2221 |
+
return null;'),
|
| 2222 |
+
new CallbackParam(), $param
|
| 2223 |
+
);
|
| 2224 |
+
$this->elements = $mapped->elements;
|
| 2225 |
+
break;
|
| 2226 |
+
default:
|
| 2227 |
+
$this->debug("Unknown pseudoclass '{$class}', skipping...");
|
| 2228 |
+
}
|
| 2229 |
+
}
|
| 2230 |
+
/**
|
| 2231 |
+
* @access private
|
| 2232 |
+
*/
|
| 2233 |
+
protected function __pseudoClassParam($paramsString) {
|
| 2234 |
+
// TODO;
|
| 2235 |
+
}
|
| 2236 |
+
/**
|
| 2237 |
+
* Enter description here...
|
| 2238 |
+
*
|
| 2239 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2240 |
+
*/
|
| 2241 |
+
public function is($selector, $nodes = null) {
|
| 2242 |
+
phpQuery::debug(array("Is:", $selector));
|
| 2243 |
+
if (! $selector)
|
| 2244 |
+
return false;
|
| 2245 |
+
$oldStack = $this->elements;
|
| 2246 |
+
$returnArray = false;
|
| 2247 |
+
if ($nodes && is_array($nodes)) {
|
| 2248 |
+
$this->elements = $nodes;
|
| 2249 |
+
} else if ($nodes)
|
| 2250 |
+
$this->elements = array($nodes);
|
| 2251 |
+
$this->filter($selector, true);
|
| 2252 |
+
$stack = $this->elements;
|
| 2253 |
+
$this->elements = $oldStack;
|
| 2254 |
+
if ($nodes)
|
| 2255 |
+
return $stack ? $stack : null;
|
| 2256 |
+
return (bool)count($stack);
|
| 2257 |
+
}
|
| 2258 |
+
/**
|
| 2259 |
+
* Enter description here...
|
| 2260 |
+
* jQuery difference.
|
| 2261 |
+
*
|
| 2262 |
+
* Callback:
|
| 2263 |
+
* - $index int
|
| 2264 |
+
* - $node DOMNode
|
| 2265 |
+
*
|
| 2266 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2267 |
+
* @link http://docs.jquery.com/Traversing/filter
|
| 2268 |
+
*/
|
| 2269 |
+
public function filterCallback($callback, $_skipHistory = false) {
|
| 2270 |
+
if (! $_skipHistory) {
|
| 2271 |
+
$this->elementsBackup = $this->elements;
|
| 2272 |
+
$this->debug("Filtering by callback");
|
| 2273 |
+
}
|
| 2274 |
+
$newStack = array();
|
| 2275 |
+
foreach($this->elements as $index => $node) {
|
| 2276 |
+
$result = phpQuery::callbackRun($callback, array($index, $node));
|
| 2277 |
+
if (is_null($result) || (! is_null($result) && $result))
|
| 2278 |
+
$newStack[] = $node;
|
| 2279 |
+
}
|
| 2280 |
+
$this->elements = $newStack;
|
| 2281 |
+
return $_skipHistory
|
| 2282 |
+
? $this
|
| 2283 |
+
: $this->newInstance();
|
| 2284 |
+
}
|
| 2285 |
+
/**
|
| 2286 |
+
* Enter description here...
|
| 2287 |
+
*
|
| 2288 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2289 |
+
* @link http://docs.jquery.com/Traversing/filter
|
| 2290 |
+
*/
|
| 2291 |
+
public function filter($selectors, $_skipHistory = false) {
|
| 2292 |
+
if ($selectors instanceof Callback OR $selectors instanceof Closure)
|
| 2293 |
+
return $this->filterCallback($selectors, $_skipHistory);
|
| 2294 |
+
if (! $_skipHistory)
|
| 2295 |
+
$this->elementsBackup = $this->elements;
|
| 2296 |
+
$notSimpleSelector = array(' ', '>', '~', '+', '/');
|
| 2297 |
+
if (! is_array($selectors))
|
| 2298 |
+
$selectors = $this->parseSelector($selectors);
|
| 2299 |
+
if (! $_skipHistory)
|
| 2300 |
+
$this->debug(array("Filtering:", $selectors));
|
| 2301 |
+
$finalStack = array();
|
| 2302 |
+
foreach($selectors as $selector) {
|
| 2303 |
+
$stack = array();
|
| 2304 |
+
if (! $selector)
|
| 2305 |
+
break;
|
| 2306 |
+
// avoid first space or /
|
| 2307 |
+
if (in_array($selector[0], $notSimpleSelector))
|
| 2308 |
+
$selector = array_slice($selector, 1);
|
| 2309 |
+
// PER NODE selector chunks
|
| 2310 |
+
foreach($this->stack() as $node) {
|
| 2311 |
+
$break = false;
|
| 2312 |
+
foreach($selector as $s) {
|
| 2313 |
+
if (!($node instanceof DOMELEMENT)) {
|
| 2314 |
+
// all besides DOMElement
|
| 2315 |
+
if ( $s[0] == '[') {
|
| 2316 |
+
$attr = trim($s, '[]');
|
| 2317 |
+
if ( mb_strpos($attr, '=')) {
|
| 2318 |
+
list( $attr, $val ) = explode('=', $attr);
|
| 2319 |
+
if ($attr == 'nodeType' && $node->nodeType != $val)
|
| 2320 |
+
$break = true;
|
| 2321 |
+
}
|
| 2322 |
+
} else
|
| 2323 |
+
$break = true;
|
| 2324 |
+
} else {
|
| 2325 |
+
// DOMElement only
|
| 2326 |
+
// ID
|
| 2327 |
+
if ( $s[0] == '#') {
|
| 2328 |
+
if ( $node->getAttribute('id') != substr($s, 1) )
|
| 2329 |
+
$break = true;
|
| 2330 |
+
// CLASSES
|
| 2331 |
+
} else if ( $s[0] == '.') {
|
| 2332 |
+
if (! $this->matchClasses( $s, $node ) )
|
| 2333 |
+
$break = true;
|
| 2334 |
+
// ATTRS
|
| 2335 |
+
} else if ( $s[0] == '[') {
|
| 2336 |
+
// strip side brackets
|
| 2337 |
+
$attr = trim($s, '[]');
|
| 2338 |
+
if (mb_strpos($attr, '=')) {
|
| 2339 |
+
list($attr, $val) = explode('=', $attr);
|
| 2340 |
+
$val = self::unQuote($val);
|
| 2341 |
+
if ($attr == 'nodeType') {
|
| 2342 |
+
if ($val != $node->nodeType)
|
| 2343 |
+
$break = true;
|
| 2344 |
+
} else if ($this->isRegexp($attr)) {
|
| 2345 |
+
$val = extension_loaded('mbstring') && phpQuery::$mbstringSupport
|
| 2346 |
+
? quotemeta(trim($val, '"\''))
|
| 2347 |
+
: preg_quote(trim($val, '"\''), '@');
|
| 2348 |
+
// switch last character
|
| 2349 |
+
switch( substr($attr, -1)) {
|
| 2350 |
+
// quotemeta used insted of preg_quote
|
| 2351 |
+
// http://code.google.com/p/phpquery/issues/detail?id=76
|
| 2352 |
+
case '^':
|
| 2353 |
+
$pattern = '^'.$val;
|
| 2354 |
+
break;
|
| 2355 |
+
case '*':
|
| 2356 |
+
$pattern = '.*'.$val.'.*';
|
| 2357 |
+
break;
|
| 2358 |
+
case '$':
|
| 2359 |
+
$pattern = '.*'.$val.'$';
|
| 2360 |
+
break;
|
| 2361 |
+
}
|
| 2362 |
+
// cut last character
|
| 2363 |
+
$attr = substr($attr, 0, -1);
|
| 2364 |
+
$isMatch = extension_loaded('mbstring') && phpQuery::$mbstringSupport
|
| 2365 |
+
? mb_ereg_match($pattern, $node->getAttribute($attr))
|
| 2366 |
+
: preg_match("@{$pattern}@", $node->getAttribute($attr));
|
| 2367 |
+
if (! $isMatch)
|
| 2368 |
+
$break = true;
|
| 2369 |
+
} else if ($node->getAttribute($attr) != $val)
|
| 2370 |
+
$break = true;
|
| 2371 |
+
} else if (! $node->hasAttribute($attr))
|
| 2372 |
+
$break = true;
|
| 2373 |
+
// PSEUDO CLASSES
|
| 2374 |
+
} else if ( $s[0] == ':') {
|
| 2375 |
+
// skip
|
| 2376 |
+
// TAG
|
| 2377 |
+
} else if (trim($s)) {
|
| 2378 |
+
if ($s != '*') {
|
| 2379 |
+
// TODO namespaces
|
| 2380 |
+
if (isset($node->tagName)) {
|
| 2381 |
+
if ($node->tagName != $s)
|
| 2382 |
+
$break = true;
|
| 2383 |
+
} else if ($s == 'html' && ! $this->isRoot($node))
|
| 2384 |
+
$break = true;
|
| 2385 |
+
}
|
| 2386 |
+
// AVOID NON-SIMPLE SELECTORS
|
| 2387 |
+
} else if (in_array($s, $notSimpleSelector)) {
|
| 2388 |
+
$break = true;
|
| 2389 |
+
$this->debug(array('Skipping non simple selector', $selector));
|
| 2390 |
+
}
|
| 2391 |
+
}
|
| 2392 |
+
if ($break)
|
| 2393 |
+
break;
|
| 2394 |
+
}
|
| 2395 |
+
// if element passed all chunks of selector - add it to new stack
|
| 2396 |
+
if (! $break )
|
| 2397 |
+
$stack[] = $node;
|
| 2398 |
+
}
|
| 2399 |
+
$tmpStack = $this->elements;
|
| 2400 |
+
$this->elements = $stack;
|
| 2401 |
+
// PER ALL NODES selector chunks
|
| 2402 |
+
foreach($selector as $s)
|
| 2403 |
+
// PSEUDO CLASSES
|
| 2404 |
+
if ($s[0] == ':')
|
| 2405 |
+
$this->pseudoClasses($s);
|
| 2406 |
+
foreach($this->elements as $node)
|
| 2407 |
+
// XXX it should be merged without duplicates
|
| 2408 |
+
// but jQuery doesnt do that
|
| 2409 |
+
$finalStack[] = $node;
|
| 2410 |
+
$this->elements = $tmpStack;
|
| 2411 |
+
}
|
| 2412 |
+
$this->elements = $finalStack;
|
| 2413 |
+
if ($_skipHistory) {
|
| 2414 |
+
return $this;
|
| 2415 |
+
} else {
|
| 2416 |
+
$this->debug("Stack length after filter(): ".count($finalStack));
|
| 2417 |
+
return $this->newInstance();
|
| 2418 |
+
}
|
| 2419 |
+
}
|
| 2420 |
+
/**
|
| 2421 |
+
*
|
| 2422 |
+
* @param $value
|
| 2423 |
+
* @return unknown_type
|
| 2424 |
+
* @TODO implement in all methods using passed parameters
|
| 2425 |
+
*/
|
| 2426 |
+
protected static function unQuote($value) {
|
| 2427 |
+
return $value[0] == '\'' || $value[0] == '"'
|
| 2428 |
+
? substr($value, 1, -1)
|
| 2429 |
+
: $value;
|
| 2430 |
+
}
|
| 2431 |
+
/**
|
| 2432 |
+
* Enter description here...
|
| 2433 |
+
*
|
| 2434 |
+
* @link http://docs.jquery.com/Ajax/load
|
| 2435 |
+
* @return phpQuery|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2436 |
+
* @todo Support $selector
|
| 2437 |
+
*/
|
| 2438 |
+
public function load($url, $data = null, $callback = null) {
|
| 2439 |
+
if ($data && ! is_array($data)) {
|
| 2440 |
+
$callback = $data;
|
| 2441 |
+
$data = null;
|
| 2442 |
+
}
|
| 2443 |
+
if (mb_strpos($url, ' ') !== false) {
|
| 2444 |
+
$matches = null;
|
| 2445 |
+
if (extension_loaded('mbstring') && phpQuery::$mbstringSupport)
|
| 2446 |
+
mb_ereg('^([^ ]+) (.*)$', $url, $matches);
|
| 2447 |
+
else
|
| 2448 |
+
preg_match('^([^ ]+) (.*)$', $url, $matches);
|
| 2449 |
+
$url = $matches[1];
|
| 2450 |
+
$selector = $matches[2];
|
| 2451 |
+
// FIXME this sucks, pass as callback param
|
| 2452 |
+
$this->_loadSelector = $selector;
|
| 2453 |
+
}
|
| 2454 |
+
$ajax = array(
|
| 2455 |
+
'url' => $url,
|
| 2456 |
+
'type' => $data ? 'POST' : 'GET',
|
| 2457 |
+
'data' => $data,
|
| 2458 |
+
'complete' => $callback,
|
| 2459 |
+
'success' => array($this, '__loadSuccess')
|
| 2460 |
+
);
|
| 2461 |
+
phpQuery::ajax($ajax);
|
| 2462 |
+
return $this;
|
| 2463 |
+
}
|
| 2464 |
+
/**
|
| 2465 |
+
* @access private
|
| 2466 |
+
* @param $html
|
| 2467 |
+
* @return unknown_type
|
| 2468 |
+
*/
|
| 2469 |
+
public function __loadSuccess($html) {
|
| 2470 |
+
if ($this->_loadSelector) {
|
| 2471 |
+
$html = phpQuery::newDocument($html)->find($this->_loadSelector);
|
| 2472 |
+
unset($this->_loadSelector);
|
| 2473 |
+
}
|
| 2474 |
+
foreach($this->stack(1) as $node) {
|
| 2475 |
+
phpQuery::pq($node, $this->getDocumentID())
|
| 2476 |
+
->markup($html);
|
| 2477 |
+
}
|
| 2478 |
+
}
|
| 2479 |
+
/**
|
| 2480 |
+
* Enter description here...
|
| 2481 |
+
*
|
| 2482 |
+
* @return phpQuery|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2483 |
+
* @todo
|
| 2484 |
+
*/
|
| 2485 |
+
public function css() {
|
| 2486 |
+
// TODO
|
| 2487 |
+
return $this;
|
| 2488 |
+
}
|
| 2489 |
+
/**
|
| 2490 |
+
* @todo
|
| 2491 |
+
*
|
| 2492 |
+
*/
|
| 2493 |
+
public function show(){
|
| 2494 |
+
// TODO
|
| 2495 |
+
return $this;
|
| 2496 |
+
}
|
| 2497 |
+
/**
|
| 2498 |
+
* @todo
|
| 2499 |
+
*
|
| 2500 |
+
*/
|
| 2501 |
+
public function hide(){
|
| 2502 |
+
// TODO
|
| 2503 |
+
return $this;
|
| 2504 |
+
}
|
| 2505 |
+
/**
|
| 2506 |
+
* Trigger a type of event on every matched element.
|
| 2507 |
+
*
|
| 2508 |
+
* @param unknown_type $type
|
| 2509 |
+
* @param unknown_type $data
|
| 2510 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2511 |
+
* @TODO support more than event in $type (space-separated)
|
| 2512 |
+
*/
|
| 2513 |
+
public function trigger($type, $data = array()) {
|
| 2514 |
+
foreach($this->elements as $node)
|
| 2515 |
+
phpQueryEvents::trigger($this->getDocumentID(), $type, $data, $node);
|
| 2516 |
+
return $this;
|
| 2517 |
+
}
|
| 2518 |
+
/**
|
| 2519 |
+
* This particular method triggers all bound event handlers on an element (for a specific event type) WITHOUT executing the browsers default actions.
|
| 2520 |
+
*
|
| 2521 |
+
* @param unknown_type $type
|
| 2522 |
+
* @param unknown_type $data
|
| 2523 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2524 |
+
* @TODO
|
| 2525 |
+
*/
|
| 2526 |
+
public function triggerHandler($type, $data = array()) {
|
| 2527 |
+
// TODO;
|
| 2528 |
+
}
|
| 2529 |
+
/**
|
| 2530 |
+
* Binds a handler to one or more events (like click) for each matched element.
|
| 2531 |
+
* Can also bind custom events.
|
| 2532 |
+
*
|
| 2533 |
+
* @param unknown_type $type
|
| 2534 |
+
* @param unknown_type $data Optional
|
| 2535 |
+
* @param unknown_type $callback
|
| 2536 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2537 |
+
* @TODO support '!' (exclusive) events
|
| 2538 |
+
* @TODO support more than event in $type (space-separated)
|
| 2539 |
+
*/
|
| 2540 |
+
public function bind($type, $data, $callback = null) {
|
| 2541 |
+
// TODO check if $data is callable, not using is_callable
|
| 2542 |
+
if (! isset($callback)) {
|
| 2543 |
+
$callback = $data;
|
| 2544 |
+
$data = null;
|
| 2545 |
+
}
|
| 2546 |
+
foreach($this->elements as $node)
|
| 2547 |
+
phpQueryEvents::add($this->getDocumentID(), $node, $type, $data, $callback);
|
| 2548 |
+
return $this;
|
| 2549 |
+
}
|
| 2550 |
+
/**
|
| 2551 |
+
* Enter description here...
|
| 2552 |
+
*
|
| 2553 |
+
* @param unknown_type $type
|
| 2554 |
+
* @param unknown_type $callback
|
| 2555 |
+
* @return unknown
|
| 2556 |
+
* @TODO namespace events
|
| 2557 |
+
* @TODO support more than event in $type (space-separated)
|
| 2558 |
+
*/
|
| 2559 |
+
public function unbind($type = null, $callback = null) {
|
| 2560 |
+
foreach($this->elements as $node)
|
| 2561 |
+
phpQueryEvents::remove($this->getDocumentID(), $node, $type, $callback);
|
| 2562 |
+
return $this;
|
| 2563 |
+
}
|
| 2564 |
+
/**
|
| 2565 |
+
* Enter description here...
|
| 2566 |
+
*
|
| 2567 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2568 |
+
*/
|
| 2569 |
+
public function change($callback = null) {
|
| 2570 |
+
if ($callback)
|
| 2571 |
+
return $this->bind('change', $callback);
|
| 2572 |
+
return $this->trigger('change');
|
| 2573 |
+
}
|
| 2574 |
+
/**
|
| 2575 |
+
* Enter description here...
|
| 2576 |
+
*
|
| 2577 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2578 |
+
*/
|
| 2579 |
+
public function submit($callback = null) {
|
| 2580 |
+
if ($callback)
|
| 2581 |
+
return $this->bind('submit', $callback);
|
| 2582 |
+
return $this->trigger('submit');
|
| 2583 |
+
}
|
| 2584 |
+
/**
|
| 2585 |
+
* Enter description here...
|
| 2586 |
+
*
|
| 2587 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2588 |
+
*/
|
| 2589 |
+
public function click($callback = null) {
|
| 2590 |
+
if ($callback)
|
| 2591 |
+
return $this->bind('click', $callback);
|
| 2592 |
+
return $this->trigger('click');
|
| 2593 |
+
}
|
| 2594 |
+
/**
|
| 2595 |
+
* Enter description here...
|
| 2596 |
+
*
|
| 2597 |
+
* @param String|phpQuery
|
| 2598 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2599 |
+
*/
|
| 2600 |
+
public function wrapAllOld($wrapper) {
|
| 2601 |
+
$wrapper = pq($wrapper)->_clone();
|
| 2602 |
+
if (! $wrapper->length() || ! $this->length() )
|
| 2603 |
+
return $this;
|
| 2604 |
+
$wrapper->insertBefore($this->elements[0]);
|
| 2605 |
+
$deepest = $wrapper->elements[0];
|
| 2606 |
+
while($deepest->firstChild && $deepest->firstChild instanceof DOMELEMENT)
|
| 2607 |
+
$deepest = $deepest->firstChild;
|
| 2608 |
+
pq($deepest)->append($this);
|
| 2609 |
+
return $this;
|
| 2610 |
+
}
|
| 2611 |
+
/**
|
| 2612 |
+
* Enter description here...
|
| 2613 |
+
*
|
| 2614 |
+
* TODO testme...
|
| 2615 |
+
* @param String|phpQuery
|
| 2616 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2617 |
+
*/
|
| 2618 |
+
public function wrapAll($wrapper) {
|
| 2619 |
+
if (! $this->length())
|
| 2620 |
+
return $this;
|
| 2621 |
+
return phpQuery::pq($wrapper, $this->getDocumentID())
|
| 2622 |
+
->clone()
|
| 2623 |
+
->insertBefore($this->get(0))
|
| 2624 |
+
->map(array($this, '___wrapAllCallback'))
|
| 2625 |
+
->append($this);
|
| 2626 |
+
}
|
| 2627 |
+
/**
|
| 2628 |
+
*
|
| 2629 |
+
* @param $node
|
| 2630 |
+
* @return unknown_type
|
| 2631 |
+
* @access private
|
| 2632 |
+
*/
|
| 2633 |
+
public function ___wrapAllCallback($node) {
|
| 2634 |
+
$deepest = $node;
|
| 2635 |
+
while($deepest->firstChild && $deepest->firstChild instanceof DOMELEMENT)
|
| 2636 |
+
$deepest = $deepest->firstChild;
|
| 2637 |
+
return $deepest;
|
| 2638 |
+
}
|
| 2639 |
+
/**
|
| 2640 |
+
* Enter description here...
|
| 2641 |
+
* NON JQUERY METHOD
|
| 2642 |
+
*
|
| 2643 |
+
* @param String|phpQuery
|
| 2644 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2645 |
+
*/
|
| 2646 |
+
public function wrapAllPHP($codeBefore, $codeAfter) {
|
| 2647 |
+
return $this
|
| 2648 |
+
->slice(0, 1)
|
| 2649 |
+
->beforePHP($codeBefore)
|
| 2650 |
+
->end()
|
| 2651 |
+
->slice(-1)
|
| 2652 |
+
->afterPHP($codeAfter)
|
| 2653 |
+
->end();
|
| 2654 |
+
}
|
| 2655 |
+
/**
|
| 2656 |
+
* Enter description here...
|
| 2657 |
+
*
|
| 2658 |
+
* @param String|phpQuery
|
| 2659 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2660 |
+
*/
|
| 2661 |
+
public function wrap($wrapper) {
|
| 2662 |
+
foreach($this->stack() as $node)
|
| 2663 |
+
phpQuery::pq($node, $this->getDocumentID())->wrapAll($wrapper);
|
| 2664 |
+
return $this;
|
| 2665 |
+
}
|
| 2666 |
+
/**
|
| 2667 |
+
* Enter description here...
|
| 2668 |
+
*
|
| 2669 |
+
* @param String|phpQuery
|
| 2670 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2671 |
+
*/
|
| 2672 |
+
public function wrapPHP($codeBefore, $codeAfter) {
|
| 2673 |
+
foreach($this->stack() as $node)
|
| 2674 |
+
phpQuery::pq($node, $this->getDocumentID())->wrapAllPHP($codeBefore, $codeAfter);
|
| 2675 |
+
return $this;
|
| 2676 |
+
}
|
| 2677 |
+
/**
|
| 2678 |
+
* Enter description here...
|
| 2679 |
+
*
|
| 2680 |
+
* @param String|phpQuery
|
| 2681 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2682 |
+
*/
|
| 2683 |
+
public function wrapInner($wrapper) {
|
| 2684 |
+
foreach($this->stack() as $node)
|
| 2685 |
+
phpQuery::pq($node, $this->getDocumentID())->contents()->wrapAll($wrapper);
|
| 2686 |
+
return $this;
|
| 2687 |
+
}
|
| 2688 |
+
/**
|
| 2689 |
+
* Enter description here...
|
| 2690 |
+
*
|
| 2691 |
+
* @param String|phpQuery
|
| 2692 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2693 |
+
*/
|
| 2694 |
+
public function wrapInnerPHP($codeBefore, $codeAfter) {
|
| 2695 |
+
foreach($this->stack(1) as $node)
|
| 2696 |
+
phpQuery::pq($node, $this->getDocumentID())->contents()
|
| 2697 |
+
->wrapAllPHP($codeBefore, $codeAfter);
|
| 2698 |
+
return $this;
|
| 2699 |
+
}
|
| 2700 |
+
/**
|
| 2701 |
+
* Enter description here...
|
| 2702 |
+
*
|
| 2703 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2704 |
+
* @testme Support for text nodes
|
| 2705 |
+
*/
|
| 2706 |
+
public function contents() {
|
| 2707 |
+
$stack = array();
|
| 2708 |
+
foreach($this->stack(1) as $el) {
|
| 2709 |
+
// FIXME (fixed) http://code.google.com/p/phpquery/issues/detail?id=56
|
| 2710 |
+
// if (! isset($el->childNodes))
|
| 2711 |
+
// continue;
|
| 2712 |
+
foreach($el->childNodes as $node) {
|
| 2713 |
+
$stack[] = $node;
|
| 2714 |
+
}
|
| 2715 |
+
}
|
| 2716 |
+
return $this->newInstance($stack);
|
| 2717 |
+
}
|
| 2718 |
+
/**
|
| 2719 |
+
* Enter description here...
|
| 2720 |
+
*
|
| 2721 |
+
* jQuery difference.
|
| 2722 |
+
*
|
| 2723 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2724 |
+
*/
|
| 2725 |
+
public function contentsUnwrap() {
|
| 2726 |
+
foreach($this->stack(1) as $node) {
|
| 2727 |
+
if (! $node->parentNode )
|
| 2728 |
+
continue;
|
| 2729 |
+
$childNodes = array();
|
| 2730 |
+
// any modification in DOM tree breaks childNodes iteration, so cache them first
|
| 2731 |
+
foreach($node->childNodes as $chNode )
|
| 2732 |
+
$childNodes[] = $chNode;
|
| 2733 |
+
foreach($childNodes as $chNode )
|
| 2734 |
+
// $node->parentNode->appendChild($chNode);
|
| 2735 |
+
$node->parentNode->insertBefore($chNode, $node);
|
| 2736 |
+
$node->parentNode->removeChild($node);
|
| 2737 |
+
}
|
| 2738 |
+
return $this;
|
| 2739 |
+
}
|
| 2740 |
+
/**
|
| 2741 |
+
* Enter description here...
|
| 2742 |
+
*
|
| 2743 |
+
* jQuery difference.
|
| 2744 |
+
*/
|
| 2745 |
+
public function switchWith($markup) {
|
| 2746 |
+
$markup = pq($markup, $this->getDocumentID());
|
| 2747 |
+
$content = null;
|
| 2748 |
+
foreach($this->stack(1) as $node) {
|
| 2749 |
+
pq($node)
|
| 2750 |
+
->contents()->toReference($content)->end()
|
| 2751 |
+
->replaceWith($markup->clone()->append($content));
|
| 2752 |
+
}
|
| 2753 |
+
return $this;
|
| 2754 |
+
}
|
| 2755 |
+
/**
|
| 2756 |
+
* Enter description here...
|
| 2757 |
+
*
|
| 2758 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2759 |
+
*/
|
| 2760 |
+
public function eq($num) {
|
| 2761 |
+
$oldStack = $this->elements;
|
| 2762 |
+
$this->elementsBackup = $this->elements;
|
| 2763 |
+
$this->elements = array();
|
| 2764 |
+
if ( isset($oldStack[$num]) )
|
| 2765 |
+
$this->elements[] = $oldStack[$num];
|
| 2766 |
+
return $this->newInstance();
|
| 2767 |
+
}
|
| 2768 |
+
/**
|
| 2769 |
+
* Enter description here...
|
| 2770 |
+
*
|
| 2771 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2772 |
+
*/
|
| 2773 |
+
public function size() {
|
| 2774 |
+
return count($this->elements);
|
| 2775 |
+
}
|
| 2776 |
+
/**
|
| 2777 |
+
* Enter description here...
|
| 2778 |
+
*
|
| 2779 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2780 |
+
* @deprecated Use length as attribute
|
| 2781 |
+
*/
|
| 2782 |
+
public function length() {
|
| 2783 |
+
return $this->size();
|
| 2784 |
+
}
|
| 2785 |
+
public function count() {
|
| 2786 |
+
return $this->size();
|
| 2787 |
+
}
|
| 2788 |
+
/**
|
| 2789 |
+
* Enter description here...
|
| 2790 |
+
*
|
| 2791 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2792 |
+
* @todo $level
|
| 2793 |
+
*/
|
| 2794 |
+
public function end($level = 1) {
|
| 2795 |
+
// $this->elements = array_pop( $this->history );
|
| 2796 |
+
// return $this;
|
| 2797 |
+
// $this->previous->DOM = $this->DOM;
|
| 2798 |
+
// $this->previous->XPath = $this->XPath;
|
| 2799 |
+
return $this->previous
|
| 2800 |
+
? $this->previous
|
| 2801 |
+
: $this;
|
| 2802 |
+
}
|
| 2803 |
+
/**
|
| 2804 |
+
* Enter description here...
|
| 2805 |
+
* Normal use ->clone() .
|
| 2806 |
+
*
|
| 2807 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2808 |
+
* @access private
|
| 2809 |
+
*/
|
| 2810 |
+
public function _clone() {
|
| 2811 |
+
$newStack = array();
|
| 2812 |
+
//pr(array('copy... ', $this->whois()));
|
| 2813 |
+
//$this->dumpHistory('copy');
|
| 2814 |
+
$this->elementsBackup = $this->elements;
|
| 2815 |
+
foreach($this->elements as $node) {
|
| 2816 |
+
$newStack[] = $node->cloneNode(true);
|
| 2817 |
+
}
|
| 2818 |
+
$this->elements = $newStack;
|
| 2819 |
+
return $this->newInstance();
|
| 2820 |
+
}
|
| 2821 |
+
/**
|
| 2822 |
+
* Enter description here...
|
| 2823 |
+
*
|
| 2824 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2825 |
+
*/
|
| 2826 |
+
public function replaceWithPHP($code) {
|
| 2827 |
+
return $this->replaceWith(phpQuery::php($code));
|
| 2828 |
+
}
|
| 2829 |
+
/**
|
| 2830 |
+
* Enter description here...
|
| 2831 |
+
*
|
| 2832 |
+
* @param String|phpQuery $content
|
| 2833 |
+
* @link http://docs.jquery.com/Manipulation/replaceWith#content
|
| 2834 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2835 |
+
*/
|
| 2836 |
+
public function replaceWith($content) {
|
| 2837 |
+
return $this->after($content)->remove();
|
| 2838 |
+
}
|
| 2839 |
+
/**
|
| 2840 |
+
* Enter description here...
|
| 2841 |
+
*
|
| 2842 |
+
* @param String $selector
|
| 2843 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2844 |
+
* @todo this works ?
|
| 2845 |
+
*/
|
| 2846 |
+
public function replaceAll($selector) {
|
| 2847 |
+
foreach(phpQuery::pq($selector, $this->getDocumentID()) as $node)
|
| 2848 |
+
phpQuery::pq($node, $this->getDocumentID())
|
| 2849 |
+
->after($this->_clone())
|
| 2850 |
+
->remove();
|
| 2851 |
+
return $this;
|
| 2852 |
+
}
|
| 2853 |
+
/**
|
| 2854 |
+
* Enter description here...
|
| 2855 |
+
*
|
| 2856 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2857 |
+
*/
|
| 2858 |
+
public function remove($selector = null) {
|
| 2859 |
+
$loop = $selector
|
| 2860 |
+
? $this->filter($selector)->elements
|
| 2861 |
+
: $this->elements;
|
| 2862 |
+
foreach($loop as $node) {
|
| 2863 |
+
if (! $node->parentNode )
|
| 2864 |
+
continue;
|
| 2865 |
+
if (isset($node->tagName))
|
| 2866 |
+
$this->debug("Removing '{$node->tagName}'");
|
| 2867 |
+
$node->parentNode->removeChild($node);
|
| 2868 |
+
// Mutation event
|
| 2869 |
+
$event = new DOMEvent(array(
|
| 2870 |
+
'target' => $node,
|
| 2871 |
+
'type' => 'DOMNodeRemoved'
|
| 2872 |
+
));
|
| 2873 |
+
phpQueryEvents::trigger($this->getDocumentID(),
|
| 2874 |
+
$event->type, array($event), $node
|
| 2875 |
+
);
|
| 2876 |
+
}
|
| 2877 |
+
return $this;
|
| 2878 |
+
}
|
| 2879 |
+
protected function markupEvents($newMarkup, $oldMarkup, $node) {
|
| 2880 |
+
if ($node->tagName == 'textarea' && $newMarkup != $oldMarkup) {
|
| 2881 |
+
$event = new DOMEvent(array(
|
| 2882 |
+
'target' => $node,
|
| 2883 |
+
'type' => 'change'
|
| 2884 |
+
));
|
| 2885 |
+
phpQueryEvents::trigger($this->getDocumentID(),
|
| 2886 |
+
$event->type, array($event), $node
|
| 2887 |
+
);
|
| 2888 |
+
}
|
| 2889 |
+
}
|
| 2890 |
+
/**
|
| 2891 |
+
* jQuey difference
|
| 2892 |
+
*
|
| 2893 |
+
* @param $markup
|
| 2894 |
+
* @return unknown_type
|
| 2895 |
+
* @TODO trigger change event for textarea
|
| 2896 |
+
*/
|
| 2897 |
+
public function markup($markup = null, $callback1 = null, $callback2 = null, $callback3 = null) {
|
| 2898 |
+
$args = func_get_args();
|
| 2899 |
+
if ($this->documentWrapper->isXML)
|
| 2900 |
+
return call_user_func_array(array($this, 'xml'), $args);
|
| 2901 |
+
else
|
| 2902 |
+
return call_user_func_array(array($this, 'html'), $args);
|
| 2903 |
+
}
|
| 2904 |
+
/**
|
| 2905 |
+
* jQuey difference
|
| 2906 |
+
*
|
| 2907 |
+
* @param $markup
|
| 2908 |
+
* @return unknown_type
|
| 2909 |
+
*/
|
| 2910 |
+
public function markupOuter($callback1 = null, $callback2 = null, $callback3 = null) {
|
| 2911 |
+
$args = func_get_args();
|
| 2912 |
+
if ($this->documentWrapper->isXML)
|
| 2913 |
+
return call_user_func_array(array($this, 'xmlOuter'), $args);
|
| 2914 |
+
else
|
| 2915 |
+
return call_user_func_array(array($this, 'htmlOuter'), $args);
|
| 2916 |
+
}
|
| 2917 |
+
/**
|
| 2918 |
+
* Enter description here...
|
| 2919 |
+
*
|
| 2920 |
+
* @param unknown_type $html
|
| 2921 |
+
* @return string|phpQuery|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2922 |
+
* @TODO force html result
|
| 2923 |
+
*/
|
| 2924 |
+
public function html($html = null, $callback1 = null, $callback2 = null, $callback3 = null) {
|
| 2925 |
+
if (isset($html)) {
|
| 2926 |
+
// INSERT
|
| 2927 |
+
$nodes = $this->documentWrapper->import($html);
|
| 2928 |
+
$this->empty();
|
| 2929 |
+
foreach($this->stack(1) as $alreadyAdded => $node) {
|
| 2930 |
+
// for now, limit events for textarea
|
| 2931 |
+
if (($this->isXHTML() || $this->isHTML()) && $node->tagName == 'textarea')
|
| 2932 |
+
$oldHtml = pq($node, $this->getDocumentID())->markup();
|
| 2933 |
+
foreach($nodes as $newNode) {
|
| 2934 |
+
$node->appendChild($alreadyAdded
|
| 2935 |
+
? $newNode->cloneNode(true)
|
| 2936 |
+
: $newNode
|
| 2937 |
+
);
|
| 2938 |
+
}
|
| 2939 |
+
// for now, limit events for textarea
|
| 2940 |
+
if (($this->isXHTML() || $this->isHTML()) && $node->tagName == 'textarea')
|
| 2941 |
+
$this->markupEvents($html, $oldHtml, $node);
|
| 2942 |
+
}
|
| 2943 |
+
return $this;
|
| 2944 |
+
} else {
|
| 2945 |
+
// FETCH
|
| 2946 |
+
$return = $this->documentWrapper->markup($this->elements, true);
|
| 2947 |
+
$args = func_get_args();
|
| 2948 |
+
foreach(array_slice($args, 1) as $callback) {
|
| 2949 |
+
$return = phpQuery::callbackRun($callback, array($return));
|
| 2950 |
+
}
|
| 2951 |
+
return $return;
|
| 2952 |
+
}
|
| 2953 |
+
}
|
| 2954 |
+
/**
|
| 2955 |
+
* @TODO force xml result
|
| 2956 |
+
*/
|
| 2957 |
+
public function xml($xml = null, $callback1 = null, $callback2 = null, $callback3 = null) {
|
| 2958 |
+
$args = func_get_args();
|
| 2959 |
+
return call_user_func_array(array($this, 'html'), $args);
|
| 2960 |
+
}
|
| 2961 |
+
/**
|
| 2962 |
+
* Enter description here...
|
| 2963 |
+
* @TODO force html result
|
| 2964 |
+
*
|
| 2965 |
+
* @return String
|
| 2966 |
+
*/
|
| 2967 |
+
public function htmlOuter($callback1 = null, $callback2 = null, $callback3 = null) {
|
| 2968 |
+
$markup = $this->documentWrapper->markup($this->elements);
|
| 2969 |
+
// pass thou callbacks
|
| 2970 |
+
$args = func_get_args();
|
| 2971 |
+
foreach($args as $callback) {
|
| 2972 |
+
$markup = phpQuery::callbackRun($callback, array($markup));
|
| 2973 |
+
}
|
| 2974 |
+
return $markup;
|
| 2975 |
+
}
|
| 2976 |
+
/**
|
| 2977 |
+
* @TODO force xml result
|
| 2978 |
+
*/
|
| 2979 |
+
public function xmlOuter($callback1 = null, $callback2 = null, $callback3 = null) {
|
| 2980 |
+
$args = func_get_args();
|
| 2981 |
+
return call_user_func_array(array($this, 'htmlOuter'), $args);
|
| 2982 |
+
}
|
| 2983 |
+
public function __toString() {
|
| 2984 |
+
return $this->markupOuter();
|
| 2985 |
+
}
|
| 2986 |
+
/**
|
| 2987 |
+
* Just like html(), but returns markup with VALID (dangerous) PHP tags.
|
| 2988 |
+
*
|
| 2989 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 2990 |
+
* @todo support returning markup with PHP tags when called without param
|
| 2991 |
+
*/
|
| 2992 |
+
public function php($code = null) {
|
| 2993 |
+
return $this->markupPHP($code);
|
| 2994 |
+
}
|
| 2995 |
+
/**
|
| 2996 |
+
* Enter description here...
|
| 2997 |
+
*
|
| 2998 |
+
* @param $code
|
| 2999 |
+
* @return unknown_type
|
| 3000 |
+
*/
|
| 3001 |
+
public function markupPHP($code = null) {
|
| 3002 |
+
return isset($code)
|
| 3003 |
+
? $this->markup(phpQuery::php($code))
|
| 3004 |
+
: phpQuery::markupToPHP($this->markup());
|
| 3005 |
+
}
|
| 3006 |
+
/**
|
| 3007 |
+
* Enter description here...
|
| 3008 |
+
*
|
| 3009 |
+
* @param $code
|
| 3010 |
+
* @return unknown_type
|
| 3011 |
+
*/
|
| 3012 |
+
public function markupOuterPHP() {
|
| 3013 |
+
return phpQuery::markupToPHP($this->markupOuter());
|
| 3014 |
+
}
|
| 3015 |
+
/**
|
| 3016 |
+
* Enter description here...
|
| 3017 |
+
*
|
| 3018 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3019 |
+
*/
|
| 3020 |
+
public function children($selector = null) {
|
| 3021 |
+
$stack = array();
|
| 3022 |
+
foreach($this->stack(1) as $node) {
|
| 3023 |
+
// foreach($node->getElementsByTagName('*') as $newNode) {
|
| 3024 |
+
foreach($node->childNodes as $newNode) {
|
| 3025 |
+
if ($newNode->nodeType != 1)
|
| 3026 |
+
continue;
|
| 3027 |
+
if ($selector && ! $this->is($selector, $newNode))
|
| 3028 |
+
continue;
|
| 3029 |
+
if ($this->elementsContainsNode($newNode, $stack))
|
| 3030 |
+
continue;
|
| 3031 |
+
$stack[] = $newNode;
|
| 3032 |
+
}
|
| 3033 |
+
}
|
| 3034 |
+
$this->elementsBackup = $this->elements;
|
| 3035 |
+
$this->elements = $stack;
|
| 3036 |
+
return $this->newInstance();
|
| 3037 |
+
}
|
| 3038 |
+
/**
|
| 3039 |
+
* Enter description here...
|
| 3040 |
+
*
|
| 3041 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3042 |
+
*/
|
| 3043 |
+
public function ancestors($selector = null) {
|
| 3044 |
+
return $this->children( $selector );
|
| 3045 |
+
}
|
| 3046 |
+
/**
|
| 3047 |
+
* Enter description here...
|
| 3048 |
+
*
|
| 3049 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3050 |
+
*/
|
| 3051 |
+
public function append( $content) {
|
| 3052 |
+
return $this->insert($content, __FUNCTION__);
|
| 3053 |
+
}
|
| 3054 |
+
/**
|
| 3055 |
+
* Enter description here...
|
| 3056 |
+
*
|
| 3057 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3058 |
+
*/
|
| 3059 |
+
public function appendPHP( $content) {
|
| 3060 |
+
return $this->insert("<php><!-- {$content} --></php>", 'append');
|
| 3061 |
+
}
|
| 3062 |
+
/**
|
| 3063 |
+
* Enter description here...
|
| 3064 |
+
*
|
| 3065 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3066 |
+
*/
|
| 3067 |
+
public function appendTo( $seletor) {
|
| 3068 |
+
return $this->insert($seletor, __FUNCTION__);
|
| 3069 |
+
}
|
| 3070 |
+
/**
|
| 3071 |
+
* Enter description here...
|
| 3072 |
+
*
|
| 3073 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3074 |
+
*/
|
| 3075 |
+
public function prepend( $content) {
|
| 3076 |
+
return $this->insert($content, __FUNCTION__);
|
| 3077 |
+
}
|
| 3078 |
+
/**
|
| 3079 |
+
* Enter description here...
|
| 3080 |
+
*
|
| 3081 |
+
* @todo accept many arguments, which are joined, arrays maybe also
|
| 3082 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3083 |
+
*/
|
| 3084 |
+
public function prependPHP( $content) {
|
| 3085 |
+
return $this->insert("<php><!-- {$content} --></php>", 'prepend');
|
| 3086 |
+
}
|
| 3087 |
+
/**
|
| 3088 |
+
* Enter description here...
|
| 3089 |
+
*
|
| 3090 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3091 |
+
*/
|
| 3092 |
+
public function prependTo( $seletor) {
|
| 3093 |
+
return $this->insert($seletor, __FUNCTION__);
|
| 3094 |
+
}
|
| 3095 |
+
/**
|
| 3096 |
+
* Enter description here...
|
| 3097 |
+
*
|
| 3098 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3099 |
+
*/
|
| 3100 |
+
public function before($content) {
|
| 3101 |
+
return $this->insert($content, __FUNCTION__);
|
| 3102 |
+
}
|
| 3103 |
+
/**
|
| 3104 |
+
* Enter description here...
|
| 3105 |
+
*
|
| 3106 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3107 |
+
*/
|
| 3108 |
+
public function beforePHP( $content) {
|
| 3109 |
+
return $this->insert("<php><!-- {$content} --></php>", 'before');
|
| 3110 |
+
}
|
| 3111 |
+
/**
|
| 3112 |
+
* Enter description here...
|
| 3113 |
+
*
|
| 3114 |
+
* @param String|phpQuery
|
| 3115 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3116 |
+
*/
|
| 3117 |
+
public function insertBefore( $seletor) {
|
| 3118 |
+
return $this->insert($seletor, __FUNCTION__);
|
| 3119 |
+
}
|
| 3120 |
+
/**
|
| 3121 |
+
* Enter description here...
|
| 3122 |
+
*
|
| 3123 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3124 |
+
*/
|
| 3125 |
+
public function after( $content) {
|
| 3126 |
+
return $this->insert($content, __FUNCTION__);
|
| 3127 |
+
}
|
| 3128 |
+
/**
|
| 3129 |
+
* Enter description here...
|
| 3130 |
+
*
|
| 3131 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3132 |
+
*/
|
| 3133 |
+
public function afterPHP( $content) {
|
| 3134 |
+
return $this->insert("<php><!-- {$content} --></php>", 'after');
|
| 3135 |
+
}
|
| 3136 |
+
/**
|
| 3137 |
+
* Enter description here...
|
| 3138 |
+
*
|
| 3139 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3140 |
+
*/
|
| 3141 |
+
public function insertAfter( $seletor) {
|
| 3142 |
+
return $this->insert($seletor, __FUNCTION__);
|
| 3143 |
+
}
|
| 3144 |
+
/**
|
| 3145 |
+
* Internal insert method. Don't use it.
|
| 3146 |
+
*
|
| 3147 |
+
* @param unknown_type $target
|
| 3148 |
+
* @param unknown_type $type
|
| 3149 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3150 |
+
* @access private
|
| 3151 |
+
*/
|
| 3152 |
+
public function insert($target, $type) {
|
| 3153 |
+
$this->debug("Inserting data with '{$type}'");
|
| 3154 |
+
$to = false;
|
| 3155 |
+
switch( $type) {
|
| 3156 |
+
case 'appendTo':
|
| 3157 |
+
case 'prependTo':
|
| 3158 |
+
case 'insertBefore':
|
| 3159 |
+
case 'insertAfter':
|
| 3160 |
+
$to = true;
|
| 3161 |
+
}
|
| 3162 |
+
switch(gettype($target)) {
|
| 3163 |
+
case 'string':
|
| 3164 |
+
$insertFrom = $insertTo = array();
|
| 3165 |
+
if ($to) {
|
| 3166 |
+
// INSERT TO
|
| 3167 |
+
$insertFrom = $this->elements;
|
| 3168 |
+
if (phpQuery::isMarkup($target)) {
|
| 3169 |
+
// $target is new markup, import it
|
| 3170 |
+
$insertTo = $this->documentWrapper->import($target);
|
| 3171 |
+
// insert into selected element
|
| 3172 |
+
} else {
|
| 3173 |
+
// $tagret is a selector
|
| 3174 |
+
$thisStack = $this->elements;
|
| 3175 |
+
$this->toRoot();
|
| 3176 |
+
$insertTo = $this->find($target)->elements;
|
| 3177 |
+
$this->elements = $thisStack;
|
| 3178 |
+
}
|
| 3179 |
+
} else {
|
| 3180 |
+
// INSERT FROM
|
| 3181 |
+
$insertTo = $this->elements;
|
| 3182 |
+
$insertFrom = $this->documentWrapper->import($target);
|
| 3183 |
+
}
|
| 3184 |
+
break;
|
| 3185 |
+
case 'object':
|
| 3186 |
+
$insertFrom = $insertTo = array();
|
| 3187 |
+
// phpQuery
|
| 3188 |
+
if ($target instanceof self) {
|
| 3189 |
+
if ($to) {
|
| 3190 |
+
$insertTo = $target->elements;
|
| 3191 |
+
if ($this->documentFragment && $this->stackIsRoot())
|
| 3192 |
+
// get all body children
|
| 3193 |
+
// $loop = $this->find('body > *')->elements;
|
| 3194 |
+
// TODO test it, test it hard...
|
| 3195 |
+
// $loop = $this->newInstance($this->root)->find('> *')->elements;
|
| 3196 |
+
$loop = $this->root->childNodes;
|
| 3197 |
+
else
|
| 3198 |
+
$loop = $this->elements;
|
| 3199 |
+
// import nodes if needed
|
| 3200 |
+
$insertFrom = $this->getDocumentID() == $target->getDocumentID()
|
| 3201 |
+
? $loop
|
| 3202 |
+
: $target->documentWrapper->import($loop);
|
| 3203 |
+
} else {
|
| 3204 |
+
$insertTo = $this->elements;
|
| 3205 |
+
if ( $target->documentFragment && $target->stackIsRoot() )
|
| 3206 |
+
// get all body children
|
| 3207 |
+
// $loop = $target->find('body > *')->elements;
|
| 3208 |
+
$loop = $target->root->childNodes;
|
| 3209 |
+
else
|
| 3210 |
+
$loop = $target->elements;
|
| 3211 |
+
// import nodes if needed
|
| 3212 |
+
$insertFrom = $this->getDocumentID() == $target->getDocumentID()
|
| 3213 |
+
? $loop
|
| 3214 |
+
: $this->documentWrapper->import($loop);
|
| 3215 |
+
}
|
| 3216 |
+
// DOMNODE
|
| 3217 |
+
} elseif ($target instanceof DOMNODE) {
|
| 3218 |
+
// import node if needed
|
| 3219 |
+
// if ( $target->ownerDocument != $this->DOM )
|
| 3220 |
+
// $target = $this->DOM->importNode($target, true);
|
| 3221 |
+
if ( $to) {
|
| 3222 |
+
$insertTo = array($target);
|
| 3223 |
+
if ($this->documentFragment && $this->stackIsRoot())
|
| 3224 |
+
// get all body children
|
| 3225 |
+
$loop = $this->root->childNodes;
|
| 3226 |
+
// $loop = $this->find('body > *')->elements;
|
| 3227 |
+
else
|
| 3228 |
+
$loop = $this->elements;
|
| 3229 |
+
foreach($loop as $fromNode)
|
| 3230 |
+
// import nodes if needed
|
| 3231 |
+
$insertFrom[] = ! $fromNode->ownerDocument->isSameNode($target->ownerDocument)
|
| 3232 |
+
? $target->ownerDocument->importNode($fromNode, true)
|
| 3233 |
+
: $fromNode;
|
| 3234 |
+
} else {
|
| 3235 |
+
// import node if needed
|
| 3236 |
+
if (! $target->ownerDocument->isSameNode($this->document))
|
| 3237 |
+
$target = $this->document->importNode($target, true);
|
| 3238 |
+
$insertTo = $this->elements;
|
| 3239 |
+
$insertFrom[] = $target;
|
| 3240 |
+
}
|
| 3241 |
+
}
|
| 3242 |
+
break;
|
| 3243 |
+
}
|
| 3244 |
+
phpQuery::debug("From ".count($insertFrom)."; To ".count($insertTo)." nodes");
|
| 3245 |
+
foreach($insertTo as $insertNumber => $toNode) {
|
| 3246 |
+
// we need static relative elements in some cases
|
| 3247 |
+
switch( $type) {
|
| 3248 |
+
case 'prependTo':
|
| 3249 |
+
case 'prepend':
|
| 3250 |
+
$firstChild = $toNode->firstChild;
|
| 3251 |
+
break;
|
| 3252 |
+
case 'insertAfter':
|
| 3253 |
+
case 'after':
|
| 3254 |
+
$nextSibling = $toNode->nextSibling;
|
| 3255 |
+
break;
|
| 3256 |
+
}
|
| 3257 |
+
foreach($insertFrom as $fromNode) {
|
| 3258 |
+
// clone if inserted already before
|
| 3259 |
+
$insert = $insertNumber
|
| 3260 |
+
? $fromNode->cloneNode(true)
|
| 3261 |
+
: $fromNode;
|
| 3262 |
+
switch($type) {
|
| 3263 |
+
case 'appendTo':
|
| 3264 |
+
case 'append':
|
| 3265 |
+
// $toNode->insertBefore(
|
| 3266 |
+
// $fromNode,
|
| 3267 |
+
// $toNode->lastChild->nextSibling
|
| 3268 |
+
// );
|
| 3269 |
+
$toNode->appendChild($insert);
|
| 3270 |
+
$eventTarget = $insert;
|
| 3271 |
+
break;
|
| 3272 |
+
case 'prependTo':
|
| 3273 |
+
case 'prepend':
|
| 3274 |
+
$toNode->insertBefore(
|
| 3275 |
+
$insert,
|
| 3276 |
+
$firstChild
|
| 3277 |
+
);
|
| 3278 |
+
break;
|
| 3279 |
+
case 'insertBefore':
|
| 3280 |
+
case 'before':
|
| 3281 |
+
if (! $toNode->parentNode)
|
| 3282 |
+
throw new Exception("No parentNode, can't do {$type}()");
|
| 3283 |
+
else
|
| 3284 |
+
$toNode->parentNode->insertBefore(
|
| 3285 |
+
$insert,
|
| 3286 |
+
$toNode
|
| 3287 |
+
);
|
| 3288 |
+
break;
|
| 3289 |
+
case 'insertAfter':
|
| 3290 |
+
case 'after':
|
| 3291 |
+
if (! $toNode->parentNode)
|
| 3292 |
+
throw new Exception("No parentNode, can't do {$type}()");
|
| 3293 |
+
else
|
| 3294 |
+
$toNode->parentNode->insertBefore(
|
| 3295 |
+
$insert,
|
| 3296 |
+
$nextSibling
|
| 3297 |
+
);
|
| 3298 |
+
break;
|
| 3299 |
+
}
|
| 3300 |
+
// Mutation event
|
| 3301 |
+
$event = new DOMEvent(array(
|
| 3302 |
+
'target' => $insert,
|
| 3303 |
+
'type' => 'DOMNodeInserted'
|
| 3304 |
+
));
|
| 3305 |
+
phpQueryEvents::trigger($this->getDocumentID(),
|
| 3306 |
+
$event->type, array($event), $insert
|
| 3307 |
+
);
|
| 3308 |
+
}
|
| 3309 |
+
}
|
| 3310 |
+
return $this;
|
| 3311 |
+
}
|
| 3312 |
+
/**
|
| 3313 |
+
* Enter description here...
|
| 3314 |
+
*
|
| 3315 |
+
* @return Int
|
| 3316 |
+
*/
|
| 3317 |
+
public function index($subject) {
|
| 3318 |
+
$index = -1;
|
| 3319 |
+
$subject = $subject instanceof phpQueryObject
|
| 3320 |
+
? $subject->elements[0]
|
| 3321 |
+
: $subject;
|
| 3322 |
+
foreach($this->newInstance() as $k => $node) {
|
| 3323 |
+
if ($node->isSameNode($subject))
|
| 3324 |
+
$index = $k;
|
| 3325 |
+
}
|
| 3326 |
+
return $index;
|
| 3327 |
+
}
|
| 3328 |
+
/**
|
| 3329 |
+
* Enter description here...
|
| 3330 |
+
*
|
| 3331 |
+
* @param unknown_type $start
|
| 3332 |
+
* @param unknown_type $end
|
| 3333 |
+
*
|
| 3334 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3335 |
+
* @testme
|
| 3336 |
+
*/
|
| 3337 |
+
public function slice($start, $end = null) {
|
| 3338 |
+
// $last = count($this->elements)-1;
|
| 3339 |
+
// $end = $end
|
| 3340 |
+
// ? min($end, $last)
|
| 3341 |
+
// : $last;
|
| 3342 |
+
// if ($start < 0)
|
| 3343 |
+
// $start = $last+$start;
|
| 3344 |
+
// if ($start > $last)
|
| 3345 |
+
// return array();
|
| 3346 |
+
if ($end > 0)
|
| 3347 |
+
$end = $end-$start;
|
| 3348 |
+
return $this->newInstance(
|
| 3349 |
+
array_slice($this->elements, $start, $end)
|
| 3350 |
+
);
|
| 3351 |
+
}
|
| 3352 |
+
/**
|
| 3353 |
+
* Enter description here...
|
| 3354 |
+
*
|
| 3355 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3356 |
+
*/
|
| 3357 |
+
public function reverse() {
|
| 3358 |
+
$this->elementsBackup = $this->elements;
|
| 3359 |
+
$this->elements = array_reverse($this->elements);
|
| 3360 |
+
return $this->newInstance();
|
| 3361 |
+
}
|
| 3362 |
+
/**
|
| 3363 |
+
* Return joined text content.
|
| 3364 |
+
* @return String
|
| 3365 |
+
*/
|
| 3366 |
+
public function text($text = null, $callback1 = null, $callback2 = null, $callback3 = null) {
|
| 3367 |
+
if (isset($text))
|
| 3368 |
+
return $this->html(htmlspecialchars($text));
|
| 3369 |
+
$args = func_get_args();
|
| 3370 |
+
$args = array_slice($args, 1);
|
| 3371 |
+
$return = '';
|
| 3372 |
+
foreach($this->elements as $node) {
|
| 3373 |
+
$text = $node->textContent;
|
| 3374 |
+
if (count($this->elements) > 1 && $text)
|
| 3375 |
+
$text .= "\n";
|
| 3376 |
+
foreach($args as $callback) {
|
| 3377 |
+
$text = phpQuery::callbackRun($callback, array($text));
|
| 3378 |
+
}
|
| 3379 |
+
$return .= $text;
|
| 3380 |
+
}
|
| 3381 |
+
return $return;
|
| 3382 |
+
}
|
| 3383 |
+
/**
|
| 3384 |
+
* Enter description here...
|
| 3385 |
+
*
|
| 3386 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3387 |
+
*/
|
| 3388 |
+
public function plugin($class, $file = null) {
|
| 3389 |
+
phpQuery::plugin($class, $file);
|
| 3390 |
+
return $this;
|
| 3391 |
+
}
|
| 3392 |
+
/**
|
| 3393 |
+
* Deprecated, use $pq->plugin() instead.
|
| 3394 |
+
*
|
| 3395 |
+
* @deprecated
|
| 3396 |
+
* @param $class
|
| 3397 |
+
* @param $file
|
| 3398 |
+
* @return unknown_type
|
| 3399 |
+
*/
|
| 3400 |
+
public static function extend($class, $file = null) {
|
| 3401 |
+
return $this->plugin($class, $file);
|
| 3402 |
+
}
|
| 3403 |
+
/**
|
| 3404 |
+
*
|
| 3405 |
+
* @access private
|
| 3406 |
+
* @param $method
|
| 3407 |
+
* @param $args
|
| 3408 |
+
* @return unknown_type
|
| 3409 |
+
*/
|
| 3410 |
+
public function __call($method, $args) {
|
| 3411 |
+
$aliasMethods = array('clone', 'empty');
|
| 3412 |
+
if (isset(phpQuery::$extendMethods[$method])) {
|
| 3413 |
+
array_unshift($args, $this);
|
| 3414 |
+
return phpQuery::callbackRun(
|
| 3415 |
+
phpQuery::$extendMethods[$method], $args
|
| 3416 |
+
);
|
| 3417 |
+
} else if (isset(phpQuery::$pluginsMethods[$method])) {
|
| 3418 |
+
array_unshift($args, $this);
|
| 3419 |
+
$class = phpQuery::$pluginsMethods[$method];
|
| 3420 |
+
$realClass = "phpQueryObjectPlugin_$class";
|
| 3421 |
+
$return = call_user_func_array(
|
| 3422 |
+
array($realClass, $method),
|
| 3423 |
+
$args
|
| 3424 |
+
);
|
| 3425 |
+
// XXX deprecate ?
|
| 3426 |
+
return is_null($return)
|
| 3427 |
+
? $this
|
| 3428 |
+
: $return;
|
| 3429 |
+
} else if (in_array($method, $aliasMethods)) {
|
| 3430 |
+
return call_user_func_array(array($this, '_'.$method), $args);
|
| 3431 |
+
} else
|
| 3432 |
+
throw new Exception("Method '{$method}' doesnt exist");
|
| 3433 |
+
}
|
| 3434 |
+
/**
|
| 3435 |
+
* Safe rename of next().
|
| 3436 |
+
*
|
| 3437 |
+
* Use it ONLY when need to call next() on an iterated object (in same time).
|
| 3438 |
+
* Normaly there is no need to do such thing ;)
|
| 3439 |
+
*
|
| 3440 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3441 |
+
* @access private
|
| 3442 |
+
*/
|
| 3443 |
+
public function _next($selector = null) {
|
| 3444 |
+
return $this->newInstance(
|
| 3445 |
+
$this->getElementSiblings('nextSibling', $selector, true)
|
| 3446 |
+
);
|
| 3447 |
+
}
|
| 3448 |
+
/**
|
| 3449 |
+
* Use prev() and next().
|
| 3450 |
+
*
|
| 3451 |
+
* @deprecated
|
| 3452 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3453 |
+
* @access private
|
| 3454 |
+
*/
|
| 3455 |
+
public function _prev($selector = null) {
|
| 3456 |
+
return $this->prev($selector);
|
| 3457 |
+
}
|
| 3458 |
+
/**
|
| 3459 |
+
* Enter description here...
|
| 3460 |
+
*
|
| 3461 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3462 |
+
*/
|
| 3463 |
+
public function prev($selector = null) {
|
| 3464 |
+
return $this->newInstance(
|
| 3465 |
+
$this->getElementSiblings('previousSibling', $selector, true)
|
| 3466 |
+
);
|
| 3467 |
+
}
|
| 3468 |
+
/**
|
| 3469 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3470 |
+
* @todo
|
| 3471 |
+
*/
|
| 3472 |
+
public function prevAll($selector = null) {
|
| 3473 |
+
return $this->newInstance(
|
| 3474 |
+
$this->getElementSiblings('previousSibling', $selector)
|
| 3475 |
+
);
|
| 3476 |
+
}
|
| 3477 |
+
/**
|
| 3478 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3479 |
+
* @todo FIXME: returns source elements insted of next siblings
|
| 3480 |
+
*/
|
| 3481 |
+
public function nextAll($selector = null) {
|
| 3482 |
+
return $this->newInstance(
|
| 3483 |
+
$this->getElementSiblings('nextSibling', $selector)
|
| 3484 |
+
);
|
| 3485 |
+
}
|
| 3486 |
+
/**
|
| 3487 |
+
* @access private
|
| 3488 |
+
*/
|
| 3489 |
+
protected function getElementSiblings($direction, $selector = null, $limitToOne = false) {
|
| 3490 |
+
$stack = array();
|
| 3491 |
+
$count = 0;
|
| 3492 |
+
foreach($this->stack() as $node) {
|
| 3493 |
+
$test = $node;
|
| 3494 |
+
while( isset($test->{$direction}) && $test->{$direction}) {
|
| 3495 |
+
$test = $test->{$direction};
|
| 3496 |
+
if (! $test instanceof DOMELEMENT)
|
| 3497 |
+
continue;
|
| 3498 |
+
$stack[] = $test;
|
| 3499 |
+
if ($limitToOne)
|
| 3500 |
+
break;
|
| 3501 |
+
}
|
| 3502 |
+
}
|
| 3503 |
+
if ($selector) {
|
| 3504 |
+
$stackOld = $this->elements;
|
| 3505 |
+
$this->elements = $stack;
|
| 3506 |
+
$stack = $this->filter($selector, true)->stack();
|
| 3507 |
+
$this->elements = $stackOld;
|
| 3508 |
+
}
|
| 3509 |
+
return $stack;
|
| 3510 |
+
}
|
| 3511 |
+
/**
|
| 3512 |
+
* Enter description here...
|
| 3513 |
+
*
|
| 3514 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3515 |
+
*/
|
| 3516 |
+
public function siblings($selector = null) {
|
| 3517 |
+
$stack = array();
|
| 3518 |
+
$siblings = array_merge(
|
| 3519 |
+
$this->getElementSiblings('previousSibling', $selector),
|
| 3520 |
+
$this->getElementSiblings('nextSibling', $selector)
|
| 3521 |
+
);
|
| 3522 |
+
foreach($siblings as $node) {
|
| 3523 |
+
if (! $this->elementsContainsNode($node, $stack))
|
| 3524 |
+
$stack[] = $node;
|
| 3525 |
+
}
|
| 3526 |
+
return $this->newInstance($stack);
|
| 3527 |
+
}
|
| 3528 |
+
/**
|
| 3529 |
+
* Enter description here...
|
| 3530 |
+
*
|
| 3531 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3532 |
+
*/
|
| 3533 |
+
public function not($selector = null) {
|
| 3534 |
+
if (is_string($selector))
|
| 3535 |
+
phpQuery::debug(array('not', $selector));
|
| 3536 |
+
else
|
| 3537 |
+
phpQuery::debug('not');
|
| 3538 |
+
$stack = array();
|
| 3539 |
+
if ($selector instanceof self || $selector instanceof DOMNODE) {
|
| 3540 |
+
foreach($this->stack() as $node) {
|
| 3541 |
+
if ($selector instanceof self) {
|
| 3542 |
+
$matchFound = false;
|
| 3543 |
+
foreach($selector->stack() as $notNode) {
|
| 3544 |
+
if ($notNode->isSameNode($node))
|
| 3545 |
+
$matchFound = true;
|
| 3546 |
+
}
|
| 3547 |
+
if (! $matchFound)
|
| 3548 |
+
$stack[] = $node;
|
| 3549 |
+
} else if ($selector instanceof DOMNODE) {
|
| 3550 |
+
if (! $selector->isSameNode($node))
|
| 3551 |
+
$stack[] = $node;
|
| 3552 |
+
} else {
|
| 3553 |
+
if (! $this->is($selector))
|
| 3554 |
+
$stack[] = $node;
|
| 3555 |
+
}
|
| 3556 |
+
}
|
| 3557 |
+
} else {
|
| 3558 |
+
$orgStack = $this->stack();
|
| 3559 |
+
$matched = $this->filter($selector, true)->stack();
|
| 3560 |
+
// $matched = array();
|
| 3561 |
+
// // simulate OR in filter() instead of AND 5y
|
| 3562 |
+
// foreach($this->parseSelector($selector) as $s) {
|
| 3563 |
+
// $matched = array_merge($matched,
|
| 3564 |
+
// $this->filter(array($s))->stack()
|
| 3565 |
+
// );
|
| 3566 |
+
// }
|
| 3567 |
+
foreach($orgStack as $node)
|
| 3568 |
+
if (! $this->elementsContainsNode($node, $matched))
|
| 3569 |
+
$stack[] = $node;
|
| 3570 |
+
}
|
| 3571 |
+
return $this->newInstance($stack);
|
| 3572 |
+
}
|
| 3573 |
+
/**
|
| 3574 |
+
* Enter description here...
|
| 3575 |
+
*
|
| 3576 |
+
* @param string|phpQueryObject
|
| 3577 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3578 |
+
*/
|
| 3579 |
+
public function add($selector = null) {
|
| 3580 |
+
if (! $selector)
|
| 3581 |
+
return $this;
|
| 3582 |
+
$stack = array();
|
| 3583 |
+
$this->elementsBackup = $this->elements;
|
| 3584 |
+
$found = phpQuery::pq($selector, $this->getDocumentID());
|
| 3585 |
+
$this->merge($found->elements);
|
| 3586 |
+
return $this->newInstance();
|
| 3587 |
+
}
|
| 3588 |
+
/**
|
| 3589 |
+
* @access private
|
| 3590 |
+
*/
|
| 3591 |
+
protected function merge() {
|
| 3592 |
+
foreach(func_get_args() as $nodes)
|
| 3593 |
+
foreach($nodes as $newNode )
|
| 3594 |
+
if (! $this->elementsContainsNode($newNode) )
|
| 3595 |
+
$this->elements[] = $newNode;
|
| 3596 |
+
}
|
| 3597 |
+
/**
|
| 3598 |
+
* @access private
|
| 3599 |
+
* TODO refactor to stackContainsNode
|
| 3600 |
+
*/
|
| 3601 |
+
protected function elementsContainsNode($nodeToCheck, $elementsStack = null) {
|
| 3602 |
+
$loop = ! is_null($elementsStack)
|
| 3603 |
+
? $elementsStack
|
| 3604 |
+
: $this->elements;
|
| 3605 |
+
foreach($loop as $node) {
|
| 3606 |
+
if ( $node->isSameNode( $nodeToCheck ) )
|
| 3607 |
+
return true;
|
| 3608 |
+
}
|
| 3609 |
+
return false;
|
| 3610 |
+
}
|
| 3611 |
+
/**
|
| 3612 |
+
* Enter description here...
|
| 3613 |
+
*
|
| 3614 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3615 |
+
*/
|
| 3616 |
+
public function parent($selector = null) {
|
| 3617 |
+
$stack = array();
|
| 3618 |
+
foreach($this->elements as $node )
|
| 3619 |
+
if ( $node->parentNode && ! $this->elementsContainsNode($node->parentNode, $stack) )
|
| 3620 |
+
$stack[] = $node->parentNode;
|
| 3621 |
+
$this->elementsBackup = $this->elements;
|
| 3622 |
+
$this->elements = $stack;
|
| 3623 |
+
if ( $selector )
|
| 3624 |
+
$this->filter($selector, true);
|
| 3625 |
+
return $this->newInstance();
|
| 3626 |
+
}
|
| 3627 |
+
/**
|
| 3628 |
+
* Enter description here...
|
| 3629 |
+
*
|
| 3630 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3631 |
+
*/
|
| 3632 |
+
public function parents($selector = null) {
|
| 3633 |
+
$stack = array();
|
| 3634 |
+
if (! $this->elements )
|
| 3635 |
+
$this->debug('parents() - stack empty');
|
| 3636 |
+
foreach($this->elements as $node) {
|
| 3637 |
+
$test = $node;
|
| 3638 |
+
while( $test->parentNode) {
|
| 3639 |
+
$test = $test->parentNode;
|
| 3640 |
+
if ($this->isRoot($test))
|
| 3641 |
+
break;
|
| 3642 |
+
if (! $this->elementsContainsNode($test, $stack)) {
|
| 3643 |
+
$stack[] = $test;
|
| 3644 |
+
continue;
|
| 3645 |
+
}
|
| 3646 |
+
}
|
| 3647 |
+
}
|
| 3648 |
+
$this->elementsBackup = $this->elements;
|
| 3649 |
+
$this->elements = $stack;
|
| 3650 |
+
if ( $selector )
|
| 3651 |
+
$this->filter($selector, true);
|
| 3652 |
+
return $this->newInstance();
|
| 3653 |
+
}
|
| 3654 |
+
/**
|
| 3655 |
+
* Internal stack iterator.
|
| 3656 |
+
*
|
| 3657 |
+
* @access private
|
| 3658 |
+
*/
|
| 3659 |
+
public function stack($nodeTypes = null) {
|
| 3660 |
+
if (!isset($nodeTypes))
|
| 3661 |
+
return $this->elements;
|
| 3662 |
+
if (!is_array($nodeTypes))
|
| 3663 |
+
$nodeTypes = array($nodeTypes);
|
| 3664 |
+
$return = array();
|
| 3665 |
+
foreach($this->elements as $node) {
|
| 3666 |
+
if (in_array($node->nodeType, $nodeTypes))
|
| 3667 |
+
$return[] = $node;
|
| 3668 |
+
}
|
| 3669 |
+
return $return;
|
| 3670 |
+
}
|
| 3671 |
+
// TODO phpdoc; $oldAttr is result of hasAttribute, before any changes
|
| 3672 |
+
protected function attrEvents($attr, $oldAttr, $oldValue, $node) {
|
| 3673 |
+
// skip events for XML documents
|
| 3674 |
+
if (! $this->isXHTML() && ! $this->isHTML())
|
| 3675 |
+
return;
|
| 3676 |
+
$event = null;
|
| 3677 |
+
// identify
|
| 3678 |
+
$isInputValue = $node->tagName == 'input'
|
| 3679 |
+
&& (
|
| 3680 |
+
in_array($node->getAttribute('type'),
|
| 3681 |
+
array('text', 'password', 'hidden'))
|
| 3682 |
+
|| !$node->getAttribute('type')
|
| 3683 |
+
);
|
| 3684 |
+
$isRadio = $node->tagName == 'input'
|
| 3685 |
+
&& $node->getAttribute('type') == 'radio';
|
| 3686 |
+
$isCheckbox = $node->tagName == 'input'
|
| 3687 |
+
&& $node->getAttribute('type') == 'checkbox';
|
| 3688 |
+
$isOption = $node->tagName == 'option';
|
| 3689 |
+
if ($isInputValue && $attr == 'value' && $oldValue != $node->getAttribute($attr)) {
|
| 3690 |
+
$event = new DOMEvent(array(
|
| 3691 |
+
'target' => $node,
|
| 3692 |
+
'type' => 'change'
|
| 3693 |
+
));
|
| 3694 |
+
} else if (($isRadio || $isCheckbox) && $attr == 'checked' && (
|
| 3695 |
+
// check
|
| 3696 |
+
(! $oldAttr && $node->hasAttribute($attr))
|
| 3697 |
+
// un-check
|
| 3698 |
+
|| (! $node->hasAttribute($attr) && $oldAttr)
|
| 3699 |
+
)) {
|
| 3700 |
+
$event = new DOMEvent(array(
|
| 3701 |
+
'target' => $node,
|
| 3702 |
+
'type' => 'change'
|
| 3703 |
+
));
|
| 3704 |
+
} else if ($isOption && $node->parentNode && $attr == 'selected' && (
|
| 3705 |
+
// select
|
| 3706 |
+
(! $oldAttr && $node->hasAttribute($attr))
|
| 3707 |
+
// un-select
|
| 3708 |
+
|| (! $node->hasAttribute($attr) && $oldAttr)
|
| 3709 |
+
)) {
|
| 3710 |
+
$event = new DOMEvent(array(
|
| 3711 |
+
'target' => $node->parentNode,
|
| 3712 |
+
'type' => 'change'
|
| 3713 |
+
));
|
| 3714 |
+
}
|
| 3715 |
+
if ($event) {
|
| 3716 |
+
phpQueryEvents::trigger($this->getDocumentID(),
|
| 3717 |
+
$event->type, array($event), $node
|
| 3718 |
+
);
|
| 3719 |
+
}
|
| 3720 |
+
}
|
| 3721 |
+
public function attr($attr = null, $value = null) {
|
| 3722 |
+
foreach($this->stack(1) as $node) {
|
| 3723 |
+
if (! is_null($value)) {
|
| 3724 |
+
$loop = $attr == '*'
|
| 3725 |
+
? $this->getNodeAttrs($node)
|
| 3726 |
+
: array($attr);
|
| 3727 |
+
foreach($loop as $a) {
|
| 3728 |
+
$oldValue = $node->getAttribute($a);
|
| 3729 |
+
$oldAttr = $node->hasAttribute($a);
|
| 3730 |
+
// TODO raises an error when charset other than UTF-8
|
| 3731 |
+
// while document's charset is also not UTF-8
|
| 3732 |
+
@$node->setAttribute($a, $value);
|
| 3733 |
+
$this->attrEvents($a, $oldAttr, $oldValue, $node);
|
| 3734 |
+
}
|
| 3735 |
+
} else if ($attr == '*') {
|
| 3736 |
+
// jQuery difference
|
| 3737 |
+
$return = array();
|
| 3738 |
+
foreach($node->attributes as $n => $v)
|
| 3739 |
+
$return[$n] = $v->value;
|
| 3740 |
+
return $return;
|
| 3741 |
+
} else
|
| 3742 |
+
return $node->hasAttribute($attr)
|
| 3743 |
+
? $node->getAttribute($attr)
|
| 3744 |
+
: null;
|
| 3745 |
+
}
|
| 3746 |
+
return is_null($value)
|
| 3747 |
+
? '' : $this;
|
| 3748 |
+
}
|
| 3749 |
+
/**
|
| 3750 |
+
* @access private
|
| 3751 |
+
*/
|
| 3752 |
+
protected function getNodeAttrs($node) {
|
| 3753 |
+
$return = array();
|
| 3754 |
+
foreach($node->attributes as $n => $o)
|
| 3755 |
+
$return[] = $n;
|
| 3756 |
+
return $return;
|
| 3757 |
+
}
|
| 3758 |
+
/**
|
| 3759 |
+
* Enter description here...
|
| 3760 |
+
*
|
| 3761 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3762 |
+
* @todo check CDATA ???
|
| 3763 |
+
*/
|
| 3764 |
+
public function attrPHP($attr, $code) {
|
| 3765 |
+
if (! is_null($code)) {
|
| 3766 |
+
$value = '<'.'?php '.$code.' ?'.'>';
|
| 3767 |
+
// TODO tempolary solution
|
| 3768 |
+
// http://code.google.com/p/phpquery/issues/detail?id=17
|
| 3769 |
+
// if (function_exists('mb_detect_encoding') && mb_detect_encoding($value) == 'ASCII')
|
| 3770 |
+
// $value = mb_convert_encoding($value, 'UTF-8', 'HTML-ENTITIES');
|
| 3771 |
+
}
|
| 3772 |
+
foreach($this->stack(1) as $node) {
|
| 3773 |
+
if (! is_null($code)) {
|
| 3774 |
+
// $attrNode = $this->DOM->createAttribute($attr);
|
| 3775 |
+
$node->setAttribute($attr, $value);
|
| 3776 |
+
// $attrNode->value = $value;
|
| 3777 |
+
// $node->appendChild($attrNode);
|
| 3778 |
+
} else if ( $attr == '*') {
|
| 3779 |
+
// jQuery diff
|
| 3780 |
+
$return = array();
|
| 3781 |
+
foreach($node->attributes as $n => $v)
|
| 3782 |
+
$return[$n] = $v->value;
|
| 3783 |
+
return $return;
|
| 3784 |
+
} else
|
| 3785 |
+
return $node->getAttribute($attr);
|
| 3786 |
+
}
|
| 3787 |
+
return $this;
|
| 3788 |
+
}
|
| 3789 |
+
/**
|
| 3790 |
+
* Enter description here...
|
| 3791 |
+
*
|
| 3792 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3793 |
+
*/
|
| 3794 |
+
public function removeAttr($attr) {
|
| 3795 |
+
foreach($this->stack(1) as $node) {
|
| 3796 |
+
$loop = $attr == '*'
|
| 3797 |
+
? $this->getNodeAttrs($node)
|
| 3798 |
+
: array($attr);
|
| 3799 |
+
foreach($loop as $a) {
|
| 3800 |
+
$oldValue = $node->getAttribute($a);
|
| 3801 |
+
$node->removeAttribute($a);
|
| 3802 |
+
$this->attrEvents($a, $oldValue, null, $node);
|
| 3803 |
+
}
|
| 3804 |
+
}
|
| 3805 |
+
return $this;
|
| 3806 |
+
}
|
| 3807 |
+
/**
|
| 3808 |
+
* Return form element value.
|
| 3809 |
+
*
|
| 3810 |
+
* @return String Fields value.
|
| 3811 |
+
*/
|
| 3812 |
+
public function val($val = null) {
|
| 3813 |
+
if (! isset($val)) {
|
| 3814 |
+
if ($this->eq(0)->is('select')) {
|
| 3815 |
+
$selected = $this->eq(0)->find('option[selected=selected]');
|
| 3816 |
+
if ($selected->is('[value]'))
|
| 3817 |
+
return $selected->attr('value');
|
| 3818 |
+
else
|
| 3819 |
+
return $selected->text();
|
| 3820 |
+
} else if ($this->eq(0)->is('textarea'))
|
| 3821 |
+
return $this->eq(0)->markup();
|
| 3822 |
+
else
|
| 3823 |
+
return $this->eq(0)->attr('value');
|
| 3824 |
+
} else {
|
| 3825 |
+
$_val = null;
|
| 3826 |
+
foreach($this->stack(1) as $node) {
|
| 3827 |
+
$node = pq($node, $this->getDocumentID());
|
| 3828 |
+
if (is_array($val) && in_array($node->attr('type'), array('checkbox', 'radio'))) {
|
| 3829 |
+
$isChecked = in_array($node->attr('value'), $val)
|
| 3830 |
+
|| in_array($node->attr('name'), $val);
|
| 3831 |
+
if ($isChecked)
|
| 3832 |
+
$node->attr('checked', 'checked');
|
| 3833 |
+
else
|
| 3834 |
+
$node->removeAttr('checked');
|
| 3835 |
+
} else if ($node->get(0)->tagName == 'select') {
|
| 3836 |
+
if (! isset($_val)) {
|
| 3837 |
+
$_val = array();
|
| 3838 |
+
if (! is_array($val))
|
| 3839 |
+
$_val = array((string)$val);
|
| 3840 |
+
else
|
| 3841 |
+
foreach($val as $v)
|
| 3842 |
+
$_val[] = $v;
|
| 3843 |
+
}
|
| 3844 |
+
foreach($node['option']->stack(1) as $option) {
|
| 3845 |
+
$option = pq($option, $this->getDocumentID());
|
| 3846 |
+
$selected = false;
|
| 3847 |
+
// XXX: workaround for string comparsion, see issue #96
|
| 3848 |
+
// http://code.google.com/p/phpquery/issues/detail?id=96
|
| 3849 |
+
$selected = is_null($option->attr('value'))
|
| 3850 |
+
? in_array($option->markup(), $_val)
|
| 3851 |
+
: in_array($option->attr('value'), $_val);
|
| 3852 |
+
// $optionValue = $option->attr('value');
|
| 3853 |
+
// $optionText = $option->text();
|
| 3854 |
+
// $optionTextLenght = mb_strlen($optionText);
|
| 3855 |
+
// foreach($_val as $v)
|
| 3856 |
+
// if ($optionValue == $v)
|
| 3857 |
+
// $selected = true;
|
| 3858 |
+
// else if ($optionText == $v && $optionTextLenght == mb_strlen($v))
|
| 3859 |
+
// $selected = true;
|
| 3860 |
+
if ($selected)
|
| 3861 |
+
$option->attr('selected', 'selected');
|
| 3862 |
+
else
|
| 3863 |
+
$option->removeAttr('selected');
|
| 3864 |
+
}
|
| 3865 |
+
} else if ($node->get(0)->tagName == 'textarea')
|
| 3866 |
+
$node->markup($val);
|
| 3867 |
+
else
|
| 3868 |
+
$node->attr('value', $val);
|
| 3869 |
+
}
|
| 3870 |
+
}
|
| 3871 |
+
return $this;
|
| 3872 |
+
}
|
| 3873 |
+
/**
|
| 3874 |
+
* Enter description here...
|
| 3875 |
+
*
|
| 3876 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3877 |
+
*/
|
| 3878 |
+
public function andSelf() {
|
| 3879 |
+
if ( $this->previous )
|
| 3880 |
+
$this->elements = array_merge($this->elements, $this->previous->elements);
|
| 3881 |
+
return $this;
|
| 3882 |
+
}
|
| 3883 |
+
/**
|
| 3884 |
+
* Enter description here...
|
| 3885 |
+
*
|
| 3886 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3887 |
+
*/
|
| 3888 |
+
public function addClass( $className) {
|
| 3889 |
+
if (! $className)
|
| 3890 |
+
return $this;
|
| 3891 |
+
foreach($this->stack(1) as $node) {
|
| 3892 |
+
if (! $this->is(".$className", $node))
|
| 3893 |
+
$node->setAttribute(
|
| 3894 |
+
'class',
|
| 3895 |
+
trim($node->getAttribute('class').' '.$className)
|
| 3896 |
+
);
|
| 3897 |
+
}
|
| 3898 |
+
return $this;
|
| 3899 |
+
}
|
| 3900 |
+
/**
|
| 3901 |
+
* Enter description here...
|
| 3902 |
+
*
|
| 3903 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3904 |
+
*/
|
| 3905 |
+
public function addClassPHP( $className) {
|
| 3906 |
+
foreach($this->stack(1) as $node) {
|
| 3907 |
+
$classes = $node->getAttribute('class');
|
| 3908 |
+
$newValue = $classes
|
| 3909 |
+
? $classes.' <'.'?php '.$className.' ?'.'>'
|
| 3910 |
+
: '<'.'?php '.$className.' ?'.'>';
|
| 3911 |
+
$node->setAttribute('class', $newValue);
|
| 3912 |
+
}
|
| 3913 |
+
return $this;
|
| 3914 |
+
}
|
| 3915 |
+
/**
|
| 3916 |
+
* Enter description here...
|
| 3917 |
+
*
|
| 3918 |
+
* @param string $className
|
| 3919 |
+
* @return bool
|
| 3920 |
+
*/
|
| 3921 |
+
public function hasClass($className) {
|
| 3922 |
+
foreach($this->stack(1) as $node) {
|
| 3923 |
+
if ( $this->is(".$className", $node))
|
| 3924 |
+
return true;
|
| 3925 |
+
}
|
| 3926 |
+
return false;
|
| 3927 |
+
}
|
| 3928 |
+
/**
|
| 3929 |
+
* Enter description here...
|
| 3930 |
+
*
|
| 3931 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3932 |
+
*/
|
| 3933 |
+
public function removeClass($className) {
|
| 3934 |
+
foreach($this->stack(1) as $node) {
|
| 3935 |
+
$classes = explode( ' ', $node->getAttribute('class'));
|
| 3936 |
+
if ( in_array($className, $classes)) {
|
| 3937 |
+
$classes = array_diff($classes, array($className));
|
| 3938 |
+
if ( $classes )
|
| 3939 |
+
$node->setAttribute('class', implode(' ', $classes));
|
| 3940 |
+
else
|
| 3941 |
+
$node->removeAttribute('class');
|
| 3942 |
+
}
|
| 3943 |
+
}
|
| 3944 |
+
return $this;
|
| 3945 |
+
}
|
| 3946 |
+
/**
|
| 3947 |
+
* Enter description here...
|
| 3948 |
+
*
|
| 3949 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3950 |
+
*/
|
| 3951 |
+
public function toggleClass($className) {
|
| 3952 |
+
foreach($this->stack(1) as $node) {
|
| 3953 |
+
if ( $this->is( $node, '.'.$className ))
|
| 3954 |
+
$this->removeClass($className);
|
| 3955 |
+
else
|
| 3956 |
+
$this->addClass($className);
|
| 3957 |
+
}
|
| 3958 |
+
return $this;
|
| 3959 |
+
}
|
| 3960 |
+
/**
|
| 3961 |
+
* Proper name without underscore (just ->empty()) also works.
|
| 3962 |
+
*
|
| 3963 |
+
* Removes all child nodes from the set of matched elements.
|
| 3964 |
+
*
|
| 3965 |
+
* Example:
|
| 3966 |
+
* pq("p")._empty()
|
| 3967 |
+
*
|
| 3968 |
+
* HTML:
|
| 3969 |
+
* <p>Hello, <span>Person</span> <a href="#">and person</a></p>
|
| 3970 |
+
*
|
| 3971 |
+
* Result:
|
| 3972 |
+
* [ <p></p> ]
|
| 3973 |
+
*
|
| 3974 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3975 |
+
* @access private
|
| 3976 |
+
*/
|
| 3977 |
+
public function _empty() {
|
| 3978 |
+
foreach($this->stack(1) as $node) {
|
| 3979 |
+
// thx to 'dave at dgx dot cz'
|
| 3980 |
+
$node->nodeValue = '';
|
| 3981 |
+
}
|
| 3982 |
+
return $this;
|
| 3983 |
+
}
|
| 3984 |
+
/**
|
| 3985 |
+
* Enter description here...
|
| 3986 |
+
*
|
| 3987 |
+
* @param array|string $callback Expects $node as first param, $index as second
|
| 3988 |
+
* @param array $scope External variables passed to callback. Use compact('varName1', 'varName2'...) and extract($scope)
|
| 3989 |
+
* @param array $arg1 Will ba passed as third and futher args to callback.
|
| 3990 |
+
* @param array $arg2 Will ba passed as fourth and futher args to callback, and so on...
|
| 3991 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 3992 |
+
*/
|
| 3993 |
+
public function each($callback, $param1 = null, $param2 = null, $param3 = null) {
|
| 3994 |
+
$paramStructure = null;
|
| 3995 |
+
if (func_num_args() > 1) {
|
| 3996 |
+
$paramStructure = func_get_args();
|
| 3997 |
+
$paramStructure = array_slice($paramStructure, 1);
|
| 3998 |
+
}
|
| 3999 |
+
foreach($this->elements as $v)
|
| 4000 |
+
phpQuery::callbackRun($callback, array($v), $paramStructure);
|
| 4001 |
+
return $this;
|
| 4002 |
+
}
|
| 4003 |
+
/**
|
| 4004 |
+
* Run callback on actual object.
|
| 4005 |
+
*
|
| 4006 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4007 |
+
*/
|
| 4008 |
+
public function callback($callback, $param1 = null, $param2 = null, $param3 = null) {
|
| 4009 |
+
$params = func_get_args();
|
| 4010 |
+
$params[0] = $this;
|
| 4011 |
+
phpQuery::callbackRun($callback, $params);
|
| 4012 |
+
return $this;
|
| 4013 |
+
}
|
| 4014 |
+
/**
|
| 4015 |
+
* Enter description here...
|
| 4016 |
+
*
|
| 4017 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4018 |
+
* @todo add $scope and $args as in each() ???
|
| 4019 |
+
*/
|
| 4020 |
+
public function map($callback, $param1 = null, $param2 = null, $param3 = null) {
|
| 4021 |
+
// $stack = array();
|
| 4022 |
+
//// foreach($this->newInstance() as $node) {
|
| 4023 |
+
// foreach($this->newInstance() as $node) {
|
| 4024 |
+
// $result = call_user_func($callback, $node);
|
| 4025 |
+
// if ($result)
|
| 4026 |
+
// $stack[] = $result;
|
| 4027 |
+
// }
|
| 4028 |
+
$params = func_get_args();
|
| 4029 |
+
array_unshift($params, $this->elements);
|
| 4030 |
+
return $this->newInstance(
|
| 4031 |
+
call_user_func_array(array('phpQuery', 'map'), $params)
|
| 4032 |
+
// phpQuery::map($this->elements, $callback)
|
| 4033 |
+
);
|
| 4034 |
+
}
|
| 4035 |
+
/**
|
| 4036 |
+
* Enter description here...
|
| 4037 |
+
*
|
| 4038 |
+
* @param <type> $key
|
| 4039 |
+
* @param <type> $value
|
| 4040 |
+
*/
|
| 4041 |
+
public function data($key, $value = null) {
|
| 4042 |
+
if (! isset($value)) {
|
| 4043 |
+
// TODO? implement specific jQuery behavior od returning parent values
|
| 4044 |
+
// is child which we look up doesn't exist
|
| 4045 |
+
return phpQuery::data($this->get(0), $key, $value, $this->getDocumentID());
|
| 4046 |
+
} else {
|
| 4047 |
+
foreach($this as $node)
|
| 4048 |
+
phpQuery::data($node, $key, $value, $this->getDocumentID());
|
| 4049 |
+
return $this;
|
| 4050 |
+
}
|
| 4051 |
+
}
|
| 4052 |
+
/**
|
| 4053 |
+
* Enter description here...
|
| 4054 |
+
*
|
| 4055 |
+
* @param <type> $key
|
| 4056 |
+
*/
|
| 4057 |
+
public function removeData($key) {
|
| 4058 |
+
foreach($this as $node)
|
| 4059 |
+
phpQuery::removeData($node, $key, $this->getDocumentID());
|
| 4060 |
+
return $this;
|
| 4061 |
+
}
|
| 4062 |
+
// INTERFACE IMPLEMENTATIONS
|
| 4063 |
+
|
| 4064 |
+
// ITERATOR INTERFACE
|
| 4065 |
+
/**
|
| 4066 |
+
* @access private
|
| 4067 |
+
*/
|
| 4068 |
+
public function rewind(){
|
| 4069 |
+
$this->debug('iterating foreach');
|
| 4070 |
+
// phpQuery::selectDocument($this->getDocumentID());
|
| 4071 |
+
$this->elementsBackup = $this->elements;
|
| 4072 |
+
$this->elementsInterator = $this->elements;
|
| 4073 |
+
$this->valid = isset( $this->elements[0] )
|
| 4074 |
+
? 1 : 0;
|
| 4075 |
+
// $this->elements = $this->valid
|
| 4076 |
+
// ? array($this->elements[0])
|
| 4077 |
+
// : array();
|
| 4078 |
+
$this->current = 0;
|
| 4079 |
+
}
|
| 4080 |
+
/**
|
| 4081 |
+
* @access private
|
| 4082 |
+
*/
|
| 4083 |
+
public function current(){
|
| 4084 |
+
return $this->elementsInterator[ $this->current ];
|
| 4085 |
+
}
|
| 4086 |
+
/**
|
| 4087 |
+
* @access private
|
| 4088 |
+
*/
|
| 4089 |
+
public function key(){
|
| 4090 |
+
return $this->current;
|
| 4091 |
+
}
|
| 4092 |
+
/**
|
| 4093 |
+
* Double-function method.
|
| 4094 |
+
*
|
| 4095 |
+
* First: main iterator interface method.
|
| 4096 |
+
* Second: Returning next sibling, alias for _next().
|
| 4097 |
+
*
|
| 4098 |
+
* Proper functionality is choosed automagicaly.
|
| 4099 |
+
*
|
| 4100 |
+
* @see phpQueryObject::_next()
|
| 4101 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4102 |
+
*/
|
| 4103 |
+
public function next($cssSelector = null){
|
| 4104 |
+
// if ($cssSelector || $this->valid)
|
| 4105 |
+
// return $this->_next($cssSelector);
|
| 4106 |
+
$this->valid = isset( $this->elementsInterator[ $this->current+1 ] )
|
| 4107 |
+
? true
|
| 4108 |
+
: false;
|
| 4109 |
+
if (! $this->valid && $this->elementsInterator) {
|
| 4110 |
+
$this->elementsInterator = null;
|
| 4111 |
+
} else if ($this->valid) {
|
| 4112 |
+
$this->current++;
|
| 4113 |
+
} else {
|
| 4114 |
+
return $this->_next($cssSelector);
|
| 4115 |
+
}
|
| 4116 |
+
}
|
| 4117 |
+
/**
|
| 4118 |
+
* @access private
|
| 4119 |
+
*/
|
| 4120 |
+
public function valid(){
|
| 4121 |
+
return $this->valid;
|
| 4122 |
+
}
|
| 4123 |
+
// ITERATOR INTERFACE END
|
| 4124 |
+
// ARRAYACCESS INTERFACE
|
| 4125 |
+
/**
|
| 4126 |
+
* @access private
|
| 4127 |
+
*/
|
| 4128 |
+
public function offsetExists($offset) {
|
| 4129 |
+
return $this->find($offset)->size() > 0;
|
| 4130 |
+
}
|
| 4131 |
+
/**
|
| 4132 |
+
* @access private
|
| 4133 |
+
*/
|
| 4134 |
+
public function offsetGet($offset) {
|
| 4135 |
+
return $this->find($offset);
|
| 4136 |
+
}
|
| 4137 |
+
/**
|
| 4138 |
+
* @access private
|
| 4139 |
+
*/
|
| 4140 |
+
public function offsetSet($offset, $value) {
|
| 4141 |
+
// $this->find($offset)->replaceWith($value);
|
| 4142 |
+
$this->find($offset)->html($value);
|
| 4143 |
+
}
|
| 4144 |
+
/**
|
| 4145 |
+
* @access private
|
| 4146 |
+
*/
|
| 4147 |
+
public function offsetUnset($offset) {
|
| 4148 |
+
// empty
|
| 4149 |
+
throw new Exception("Can't do unset, use array interface only for calling queries and replacing HTML.");
|
| 4150 |
+
}
|
| 4151 |
+
// ARRAYACCESS INTERFACE END
|
| 4152 |
+
/**
|
| 4153 |
+
* Returns node's XPath.
|
| 4154 |
+
*
|
| 4155 |
+
* @param unknown_type $oneNode
|
| 4156 |
+
* @return string
|
| 4157 |
+
* @TODO use native getNodePath is avaible
|
| 4158 |
+
* @access private
|
| 4159 |
+
*/
|
| 4160 |
+
protected function getNodeXpath($oneNode = null, $namespace = null) {
|
| 4161 |
+
$return = array();
|
| 4162 |
+
$loop = $oneNode
|
| 4163 |
+
? array($oneNode)
|
| 4164 |
+
: $this->elements;
|
| 4165 |
+
// if ($namespace)
|
| 4166 |
+
// $namespace .= ':';
|
| 4167 |
+
foreach($loop as $node) {
|
| 4168 |
+
if ($node instanceof DOMDOCUMENT) {
|
| 4169 |
+
$return[] = '';
|
| 4170 |
+
continue;
|
| 4171 |
+
}
|
| 4172 |
+
$xpath = array();
|
| 4173 |
+
while(! ($node instanceof DOMDOCUMENT)) {
|
| 4174 |
+
$i = 1;
|
| 4175 |
+
$sibling = $node;
|
| 4176 |
+
while($sibling->previousSibling) {
|
| 4177 |
+
$sibling = $sibling->previousSibling;
|
| 4178 |
+
$isElement = $sibling instanceof DOMELEMENT;
|
| 4179 |
+
if ($isElement && $sibling->tagName == $node->tagName)
|
| 4180 |
+
$i++;
|
| 4181 |
+
}
|
| 4182 |
+
$xpath[] = $this->isXML()
|
| 4183 |
+
? "*[local-name()='{$node->tagName}'][{$i}]"
|
| 4184 |
+
: "{$node->tagName}[{$i}]";
|
| 4185 |
+
$node = $node->parentNode;
|
| 4186 |
+
}
|
| 4187 |
+
$xpath = join('/', array_reverse($xpath));
|
| 4188 |
+
$return[] = '/'.$xpath;
|
| 4189 |
+
}
|
| 4190 |
+
return $oneNode
|
| 4191 |
+
? $return[0]
|
| 4192 |
+
: $return;
|
| 4193 |
+
}
|
| 4194 |
+
// HELPERS
|
| 4195 |
+
public function whois($oneNode = null) {
|
| 4196 |
+
$return = array();
|
| 4197 |
+
$loop = $oneNode
|
| 4198 |
+
? array( $oneNode )
|
| 4199 |
+
: $this->elements;
|
| 4200 |
+
foreach($loop as $node) {
|
| 4201 |
+
if (isset($node->tagName)) {
|
| 4202 |
+
$tag = in_array($node->tagName, array('php', 'js'))
|
| 4203 |
+
? strtoupper($node->tagName)
|
| 4204 |
+
: $node->tagName;
|
| 4205 |
+
$return[] = $tag
|
| 4206 |
+
.($node->getAttribute('id')
|
| 4207 |
+
? '#'.$node->getAttribute('id'):'')
|
| 4208 |
+
.($node->getAttribute('class')
|
| 4209 |
+
? '.'.join('.', split(' ', $node->getAttribute('class'))):'')
|
| 4210 |
+
.($node->getAttribute('name')
|
| 4211 |
+
? '[name="'.$node->getAttribute('name').'"]':'')
|
| 4212 |
+
.($node->getAttribute('value') && strpos($node->getAttribute('value'), '<'.'?php') === false
|
| 4213 |
+
? '[value="'.substr(str_replace("\n", '', $node->getAttribute('value')), 0, 15).'"]':'')
|
| 4214 |
+
.($node->getAttribute('value') && strpos($node->getAttribute('value'), '<'.'?php') !== false
|
| 4215 |
+
? '[value=PHP]':'')
|
| 4216 |
+
.($node->getAttribute('selected')
|
| 4217 |
+
? '[selected]':'')
|
| 4218 |
+
.($node->getAttribute('checked')
|
| 4219 |
+
? '[checked]':'')
|
| 4220 |
+
;
|
| 4221 |
+
} else if ($node instanceof DOMTEXT) {
|
| 4222 |
+
if (trim($node->textContent))
|
| 4223 |
+
$return[] = 'Text:'.substr(str_replace("\n", ' ', $node->textContent), 0, 15);
|
| 4224 |
+
} else {
|
| 4225 |
+
|
| 4226 |
+
}
|
| 4227 |
+
}
|
| 4228 |
+
return $oneNode && isset($return[0])
|
| 4229 |
+
? $return[0]
|
| 4230 |
+
: $return;
|
| 4231 |
+
}
|
| 4232 |
+
/**
|
| 4233 |
+
* Dump htmlOuter and preserve chain. Usefull for debugging.
|
| 4234 |
+
*
|
| 4235 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4236 |
+
*
|
| 4237 |
+
*/
|
| 4238 |
+
public function dump() {
|
| 4239 |
+
print 'DUMP #'.(phpQuery::$dumpCount++).' ';
|
| 4240 |
+
$debug = phpQuery::$debug;
|
| 4241 |
+
phpQuery::$debug = false;
|
| 4242 |
+
// print __FILE__.':'.__LINE__."\n";
|
| 4243 |
+
var_dump($this->htmlOuter());
|
| 4244 |
+
return $this;
|
| 4245 |
+
}
|
| 4246 |
+
public function dumpWhois() {
|
| 4247 |
+
print 'DUMP #'.(phpQuery::$dumpCount++).' ';
|
| 4248 |
+
$debug = phpQuery::$debug;
|
| 4249 |
+
phpQuery::$debug = false;
|
| 4250 |
+
// print __FILE__.':'.__LINE__."\n";
|
| 4251 |
+
var_dump('whois', $this->whois());
|
| 4252 |
+
phpQuery::$debug = $debug;
|
| 4253 |
+
return $this;
|
| 4254 |
+
}
|
| 4255 |
+
public function dumpLength() {
|
| 4256 |
+
print 'DUMP #'.(phpQuery::$dumpCount++).' ';
|
| 4257 |
+
$debug = phpQuery::$debug;
|
| 4258 |
+
phpQuery::$debug = false;
|
| 4259 |
+
// print __FILE__.':'.__LINE__."\n";
|
| 4260 |
+
var_dump('length', $this->length());
|
| 4261 |
+
phpQuery::$debug = $debug;
|
| 4262 |
+
return $this;
|
| 4263 |
+
}
|
| 4264 |
+
public function dumpTree($html = true, $title = true) {
|
| 4265 |
+
$output = $title
|
| 4266 |
+
? 'DUMP #'.(phpQuery::$dumpCount++)." \n" : '';
|
| 4267 |
+
$debug = phpQuery::$debug;
|
| 4268 |
+
phpQuery::$debug = false;
|
| 4269 |
+
foreach($this->stack() as $node)
|
| 4270 |
+
$output .= $this->__dumpTree($node);
|
| 4271 |
+
phpQuery::$debug = $debug;
|
| 4272 |
+
print $html
|
| 4273 |
+
? nl2br(str_replace(' ', ' ', $output))
|
| 4274 |
+
: $output;
|
| 4275 |
+
return $this;
|
| 4276 |
+
}
|
| 4277 |
+
private function __dumpTree($node, $intend = 0) {
|
| 4278 |
+
$whois = $this->whois($node);
|
| 4279 |
+
$return = '';
|
| 4280 |
+
if ($whois)
|
| 4281 |
+
$return .= str_repeat(' - ', $intend).$whois."\n";
|
| 4282 |
+
if (isset($node->childNodes))
|
| 4283 |
+
foreach($node->childNodes as $chNode)
|
| 4284 |
+
$return .= $this->__dumpTree($chNode, $intend+1);
|
| 4285 |
+
return $return;
|
| 4286 |
+
}
|
| 4287 |
+
/**
|
| 4288 |
+
* Dump htmlOuter and stop script execution. Usefull for debugging.
|
| 4289 |
+
*
|
| 4290 |
+
*/
|
| 4291 |
+
public function dumpDie() {
|
| 4292 |
+
print __FILE__.':'.__LINE__;
|
| 4293 |
+
var_dump($this->htmlOuter());
|
| 4294 |
+
die();
|
| 4295 |
+
}
|
| 4296 |
+
}
|
| 4297 |
+
|
| 4298 |
+
|
| 4299 |
+
// -- Multibyte Compatibility functions ---------------------------------------
|
| 4300 |
+
// http://svn.iphonewebdev.com/lace/lib/mb_compat.php
|
| 4301 |
+
|
| 4302 |
+
/**
|
| 4303 |
+
* mb_internal_encoding()
|
| 4304 |
+
*
|
| 4305 |
+
* Included for mbstring pseudo-compatability.
|
| 4306 |
+
*/
|
| 4307 |
+
if (!function_exists('mb_internal_encoding'))
|
| 4308 |
+
{
|
| 4309 |
+
function mb_internal_encoding($enc) {return true; }
|
| 4310 |
+
}
|
| 4311 |
+
|
| 4312 |
+
/**
|
| 4313 |
+
* mb_regex_encoding()
|
| 4314 |
+
*
|
| 4315 |
+
* Included for mbstring pseudo-compatability.
|
| 4316 |
+
*/
|
| 4317 |
+
if (!function_exists('mb_regex_encoding'))
|
| 4318 |
+
{
|
| 4319 |
+
function mb_regex_encoding($enc) {return true; }
|
| 4320 |
+
}
|
| 4321 |
+
|
| 4322 |
+
/**
|
| 4323 |
+
* mb_strlen()
|
| 4324 |
+
*
|
| 4325 |
+
* Included for mbstring pseudo-compatability.
|
| 4326 |
+
*/
|
| 4327 |
+
if (!function_exists('mb_strlen'))
|
| 4328 |
+
{
|
| 4329 |
+
function mb_strlen($str)
|
| 4330 |
+
{
|
| 4331 |
+
return strlen($str);
|
| 4332 |
+
}
|
| 4333 |
+
}
|
| 4334 |
+
|
| 4335 |
+
/**
|
| 4336 |
+
* mb_strpos()
|
| 4337 |
+
*
|
| 4338 |
+
* Included for mbstring pseudo-compatability.
|
| 4339 |
+
*/
|
| 4340 |
+
if (!function_exists('mb_strpos'))
|
| 4341 |
+
{
|
| 4342 |
+
function mb_strpos($haystack, $needle, $offset=0)
|
| 4343 |
+
{
|
| 4344 |
+
return strpos($haystack, $needle, $offset);
|
| 4345 |
+
}
|
| 4346 |
+
}
|
| 4347 |
+
/**
|
| 4348 |
+
* mb_stripos()
|
| 4349 |
+
*
|
| 4350 |
+
* Included for mbstring pseudo-compatability.
|
| 4351 |
+
*/
|
| 4352 |
+
if (!function_exists('mb_stripos'))
|
| 4353 |
+
{
|
| 4354 |
+
function mb_stripos($haystack, $needle, $offset=0)
|
| 4355 |
+
{
|
| 4356 |
+
return stripos($haystack, $needle, $offset);
|
| 4357 |
+
}
|
| 4358 |
+
}
|
| 4359 |
+
|
| 4360 |
+
/**
|
| 4361 |
+
* mb_substr()
|
| 4362 |
+
*
|
| 4363 |
+
* Included for mbstring pseudo-compatability.
|
| 4364 |
+
*/
|
| 4365 |
+
if (!function_exists('mb_substr'))
|
| 4366 |
+
{
|
| 4367 |
+
function mb_substr($str, $start, $length=0)
|
| 4368 |
+
{
|
| 4369 |
+
return substr($str, $start, $length);
|
| 4370 |
+
}
|
| 4371 |
+
}
|
| 4372 |
+
|
| 4373 |
+
/**
|
| 4374 |
+
* mb_substr_count()
|
| 4375 |
+
*
|
| 4376 |
+
* Included for mbstring pseudo-compatability.
|
| 4377 |
+
*/
|
| 4378 |
+
if (!function_exists('mb_substr_count'))
|
| 4379 |
+
{
|
| 4380 |
+
function mb_substr_count($haystack, $needle)
|
| 4381 |
+
{
|
| 4382 |
+
return substr_count($haystack, $needle);
|
| 4383 |
+
}
|
| 4384 |
+
}
|
| 4385 |
+
|
| 4386 |
+
|
| 4387 |
+
/**
|
| 4388 |
+
* Static namespace for phpQuery functions.
|
| 4389 |
+
*
|
| 4390 |
+
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 4391 |
+
* @package phpQuery
|
| 4392 |
+
*/
|
| 4393 |
+
abstract class phpQuery {
|
| 4394 |
+
/**
|
| 4395 |
+
* XXX: Workaround for mbstring problems
|
| 4396 |
+
*
|
| 4397 |
+
* @var bool
|
| 4398 |
+
*/
|
| 4399 |
+
public static $mbstringSupport = true;
|
| 4400 |
+
public static $debug = false;
|
| 4401 |
+
public static $documents = array();
|
| 4402 |
+
public static $defaultDocumentID = null;
|
| 4403 |
+
// public static $defaultDoctype = 'html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"';
|
| 4404 |
+
/**
|
| 4405 |
+
* Applies only to HTML.
|
| 4406 |
+
*
|
| 4407 |
+
* @var unknown_type
|
| 4408 |
+
*/
|
| 4409 |
+
public static $defaultDoctype = '<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"
|
| 4410 |
+
"http://www.w3.org/TR/html4/loose.dtd">';
|
| 4411 |
+
public static $defaultCharset = 'UTF-8';
|
| 4412 |
+
/**
|
| 4413 |
+
* Static namespace for plugins.
|
| 4414 |
+
*
|
| 4415 |
+
* @var object
|
| 4416 |
+
*/
|
| 4417 |
+
public static $plugins = array();
|
| 4418 |
+
/**
|
| 4419 |
+
* List of loaded plugins.
|
| 4420 |
+
*
|
| 4421 |
+
* @var unknown_type
|
| 4422 |
+
*/
|
| 4423 |
+
public static $pluginsLoaded = array();
|
| 4424 |
+
public static $pluginsMethods = array();
|
| 4425 |
+
public static $pluginsStaticMethods = array();
|
| 4426 |
+
public static $extendMethods = array();
|
| 4427 |
+
/**
|
| 4428 |
+
* @TODO implement
|
| 4429 |
+
*/
|
| 4430 |
+
public static $extendStaticMethods = array();
|
| 4431 |
+
/**
|
| 4432 |
+
* Hosts allowed for AJAX connections.
|
| 4433 |
+
* Dot '.' means $_SERVER['HTTP_HOST'] (if any).
|
| 4434 |
+
*
|
| 4435 |
+
* @var array
|
| 4436 |
+
*/
|
| 4437 |
+
public static $ajaxAllowedHosts = array(
|
| 4438 |
+
'.'
|
| 4439 |
+
);
|
| 4440 |
+
/**
|
| 4441 |
+
* AJAX settings.
|
| 4442 |
+
*
|
| 4443 |
+
* @var array
|
| 4444 |
+
* XXX should it be static or not ?
|
| 4445 |
+
*/
|
| 4446 |
+
public static $ajaxSettings = array(
|
| 4447 |
+
'url' => '',//TODO
|
| 4448 |
+
'global' => true,
|
| 4449 |
+
'type' => "GET",
|
| 4450 |
+
'timeout' => null,
|
| 4451 |
+
'contentType' => "application/x-www-form-urlencoded",
|
| 4452 |
+
'processData' => true,
|
| 4453 |
+
// 'async' => true,
|
| 4454 |
+
'data' => null,
|
| 4455 |
+
'username' => null,
|
| 4456 |
+
'password' => null,
|
| 4457 |
+
'accepts' => array(
|
| 4458 |
+
'xml' => "application/xml, text/xml",
|
| 4459 |
+
'html' => "text/html",
|
| 4460 |
+
'script' => "text/javascript, application/javascript",
|
| 4461 |
+
'json' => "application/json, text/javascript",
|
| 4462 |
+
'text' => "text/plain",
|
| 4463 |
+
'_default' => "*/*"
|
| 4464 |
+
)
|
| 4465 |
+
);
|
| 4466 |
+
public static $lastModified = null;
|
| 4467 |
+
public static $active = 0;
|
| 4468 |
+
public static $dumpCount = 0;
|
| 4469 |
+
/**
|
| 4470 |
+
* Multi-purpose function.
|
| 4471 |
+
* Use pq() as shortcut.
|
| 4472 |
+
*
|
| 4473 |
+
* In below examples, $pq is any result of pq(); function.
|
| 4474 |
+
*
|
| 4475 |
+
* 1. Import markup into existing document (without any attaching):
|
| 4476 |
+
* - Import into selected document:
|
| 4477 |
+
* pq('<div/>') // DOESNT accept text nodes at beginning of input string !
|
| 4478 |
+
* - Import into document with ID from $pq->getDocumentID():
|
| 4479 |
+
* pq('<div/>', $pq->getDocumentID())
|
| 4480 |
+
* - Import into same document as DOMNode belongs to:
|
| 4481 |
+
* pq('<div/>', DOMNode)
|
| 4482 |
+
* - Import into document from phpQuery object:
|
| 4483 |
+
* pq('<div/>', $pq)
|
| 4484 |
+
*
|
| 4485 |
+
* 2. Run query:
|
| 4486 |
+
* - Run query on last selected document:
|
| 4487 |
+
* pq('div.myClass')
|
| 4488 |
+
* - Run query on document with ID from $pq->getDocumentID():
|
| 4489 |
+
* pq('div.myClass', $pq->getDocumentID())
|
| 4490 |
+
* - Run query on same document as DOMNode belongs to and use node(s)as root for query:
|
| 4491 |
+
* pq('div.myClass', DOMNode)
|
| 4492 |
+
* - Run query on document from phpQuery object
|
| 4493 |
+
* and use object's stack as root node(s) for query:
|
| 4494 |
+
* pq('div.myClass', $pq)
|
| 4495 |
+
*
|
| 4496 |
+
* @param string|DOMNode|DOMNodeList|array $arg1 HTML markup, CSS Selector, DOMNode or array of DOMNodes
|
| 4497 |
+
* @param string|phpQueryObject|DOMNode $context DOM ID from $pq->getDocumentID(), phpQuery object (determines also query root) or DOMNode (determines also query root)
|
| 4498 |
+
*
|
| 4499 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery|QueryTemplatesPhpQuery|false
|
| 4500 |
+
* phpQuery object or false in case of error.
|
| 4501 |
+
*/
|
| 4502 |
+
public static function pq($arg1, $context = null) {
|
| 4503 |
+
if ($arg1 instanceof DOMNODE && ! isset($context)) {
|
| 4504 |
+
foreach(phpQuery::$documents as $documentWrapper) {
|
| 4505 |
+
$compare = $arg1 instanceof DOMDocument
|
| 4506 |
+
? $arg1 : $arg1->ownerDocument;
|
| 4507 |
+
if ($documentWrapper->document->isSameNode($compare))
|
| 4508 |
+
$context = $documentWrapper->id;
|
| 4509 |
+
}
|
| 4510 |
+
}
|
| 4511 |
+
if (! $context) {
|
| 4512 |
+
$domId = self::$defaultDocumentID;
|
| 4513 |
+
if (! $domId)
|
| 4514 |
+
throw new Exception("Can't use last created DOM, because there isn't any. Use phpQuery::newDocument() first.");
|
| 4515 |
+
// } else if (is_object($context) && ($context instanceof PHPQUERY || is_subclass_of($context, 'phpQueryObject')))
|
| 4516 |
+
} else if (is_object($context) && $context instanceof phpQueryObject)
|
| 4517 |
+
$domId = $context->getDocumentID();
|
| 4518 |
+
else if ($context instanceof DOMDOCUMENT) {
|
| 4519 |
+
$domId = self::getDocumentID($context);
|
| 4520 |
+
if (! $domId) {
|
| 4521 |
+
//throw new Exception('Orphaned DOMDocument');
|
| 4522 |
+
$domId = self::newDocument($context)->getDocumentID();
|
| 4523 |
+
}
|
| 4524 |
+
} else if ($context instanceof DOMNODE) {
|
| 4525 |
+
$domId = self::getDocumentID($context);
|
| 4526 |
+
if (! $domId) {
|
| 4527 |
+
throw new Exception('Orphaned DOMNode');
|
| 4528 |
+
// $domId = self::newDocument($context->ownerDocument);
|
| 4529 |
+
}
|
| 4530 |
+
} else
|
| 4531 |
+
$domId = $context;
|
| 4532 |
+
if ($arg1 instanceof phpQueryObject) {
|
| 4533 |
+
// if (is_object($arg1) && (get_class($arg1) == 'phpQueryObject' || $arg1 instanceof PHPQUERY || is_subclass_of($arg1, 'phpQueryObject'))) {
|
| 4534 |
+
/**
|
| 4535 |
+
* Return $arg1 or import $arg1 stack if document differs:
|
| 4536 |
+
* pq(pq('<div/>'))
|
| 4537 |
+
*/
|
| 4538 |
+
if ($arg1->getDocumentID() == $domId)
|
| 4539 |
+
return $arg1;
|
| 4540 |
+
$class = get_class($arg1);
|
| 4541 |
+
// support inheritance by passing old object to overloaded constructor
|
| 4542 |
+
$phpQuery = $class != 'phpQuery'
|
| 4543 |
+
? new $class($arg1, $domId)
|
| 4544 |
+
: new phpQueryObject($domId);
|
| 4545 |
+
$phpQuery->elements = array();
|
| 4546 |
+
foreach($arg1->elements as $node)
|
| 4547 |
+
$phpQuery->elements[] = $phpQuery->document->importNode($node, true);
|
| 4548 |
+
return $phpQuery;
|
| 4549 |
+
} else if ($arg1 instanceof DOMNODE || (is_array($arg1) && isset($arg1[0]) && $arg1[0] instanceof DOMNODE)) {
|
| 4550 |
+
/*
|
| 4551 |
+
* Wrap DOM nodes with phpQuery object, import into document when needed:
|
| 4552 |
+
* pq(array($domNode1, $domNode2))
|
| 4553 |
+
*/
|
| 4554 |
+
$phpQuery = new phpQueryObject($domId);
|
| 4555 |
+
if (!($arg1 instanceof DOMNODELIST) && ! is_array($arg1))
|
| 4556 |
+
$arg1 = array($arg1);
|
| 4557 |
+
$phpQuery->elements = array();
|
| 4558 |
+
foreach($arg1 as $node) {
|
| 4559 |
+
$sameDocument = $node->ownerDocument instanceof DOMDOCUMENT
|
| 4560 |
+
&& ! $node->ownerDocument->isSameNode($phpQuery->document);
|
| 4561 |
+
$phpQuery->elements[] = $sameDocument
|
| 4562 |
+
? $phpQuery->document->importNode($node, true)
|
| 4563 |
+
: $node;
|
| 4564 |
+
}
|
| 4565 |
+
return $phpQuery;
|
| 4566 |
+
} else if (self::isMarkup($arg1)) {
|
| 4567 |
+
/**
|
| 4568 |
+
* Import HTML:
|
| 4569 |
+
* pq('<div/>')
|
| 4570 |
+
*/
|
| 4571 |
+
$phpQuery = new phpQueryObject($domId);
|
| 4572 |
+
return $phpQuery->newInstance(
|
| 4573 |
+
$phpQuery->documentWrapper->import($arg1)
|
| 4574 |
+
);
|
| 4575 |
+
} else {
|
| 4576 |
+
/**
|
| 4577 |
+
* Run CSS query:
|
| 4578 |
+
* pq('div.myClass')
|
| 4579 |
+
*/
|
| 4580 |
+
$phpQuery = new phpQueryObject($domId);
|
| 4581 |
+
// if ($context && ($context instanceof PHPQUERY || is_subclass_of($context, 'phpQueryObject')))
|
| 4582 |
+
if ($context && $context instanceof phpQueryObject)
|
| 4583 |
+
$phpQuery->elements = $context->elements;
|
| 4584 |
+
else if ($context && $context instanceof DOMNODELIST) {
|
| 4585 |
+
$phpQuery->elements = array();
|
| 4586 |
+
foreach($context as $node)
|
| 4587 |
+
$phpQuery->elements[] = $node;
|
| 4588 |
+
} else if ($context && $context instanceof DOMNODE)
|
| 4589 |
+
$phpQuery->elements = array($context);
|
| 4590 |
+
return $phpQuery->find($arg1);
|
| 4591 |
+
}
|
| 4592 |
+
}
|
| 4593 |
+
/**
|
| 4594 |
+
* Sets default document to $id. Document has to be loaded prior
|
| 4595 |
+
* to using this method.
|
| 4596 |
+
* $id can be retrived via getDocumentID() or getDocumentIDRef().
|
| 4597 |
+
*
|
| 4598 |
+
* @param unknown_type $id
|
| 4599 |
+
*/
|
| 4600 |
+
public static function selectDocument($id) {
|
| 4601 |
+
$id = self::getDocumentID($id);
|
| 4602 |
+
self::debug("Selecting document '$id' as default one");
|
| 4603 |
+
self::$defaultDocumentID = self::getDocumentID($id);
|
| 4604 |
+
}
|
| 4605 |
+
/**
|
| 4606 |
+
* Returns document with id $id or last used as phpQueryObject.
|
| 4607 |
+
* $id can be retrived via getDocumentID() or getDocumentIDRef().
|
| 4608 |
+
* Chainable.
|
| 4609 |
+
*
|
| 4610 |
+
* @see phpQuery::selectDocument()
|
| 4611 |
+
* @param unknown_type $id
|
| 4612 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4613 |
+
*/
|
| 4614 |
+
public static function getDocument($id = null) {
|
| 4615 |
+
if ($id)
|
| 4616 |
+
phpQuery::selectDocument($id);
|
| 4617 |
+
else
|
| 4618 |
+
$id = phpQuery::$defaultDocumentID;
|
| 4619 |
+
return new phpQueryObject($id);
|
| 4620 |
+
}
|
| 4621 |
+
/**
|
| 4622 |
+
* Creates new document from markup.
|
| 4623 |
+
* Chainable.
|
| 4624 |
+
*
|
| 4625 |
+
* @param unknown_type $markup
|
| 4626 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4627 |
+
*/
|
| 4628 |
+
public static function newDocument($markup = null, $contentType = null) {
|
| 4629 |
+
if (! $markup)
|
| 4630 |
+
$markup = '';
|
| 4631 |
+
$documentID = phpQuery::createDocumentWrapper($markup, $contentType);
|
| 4632 |
+
return new phpQueryObject($documentID);
|
| 4633 |
+
}
|
| 4634 |
+
/**
|
| 4635 |
+
* Creates new document from markup.
|
| 4636 |
+
* Chainable.
|
| 4637 |
+
*
|
| 4638 |
+
* @param unknown_type $markup
|
| 4639 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4640 |
+
*/
|
| 4641 |
+
public static function newDocumentHTML($markup = null, $charset = null) {
|
| 4642 |
+
$contentType = $charset
|
| 4643 |
+
? ";charset=$charset"
|
| 4644 |
+
: '';
|
| 4645 |
+
return self::newDocument($markup, "text/html{$contentType}");
|
| 4646 |
+
}
|
| 4647 |
+
/**
|
| 4648 |
+
* Creates new document from markup.
|
| 4649 |
+
* Chainable.
|
| 4650 |
+
*
|
| 4651 |
+
* @param unknown_type $markup
|
| 4652 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4653 |
+
*/
|
| 4654 |
+
public static function newDocumentXML($markup = null, $charset = null) {
|
| 4655 |
+
$contentType = $charset
|
| 4656 |
+
? ";charset=$charset"
|
| 4657 |
+
: '';
|
| 4658 |
+
return self::newDocument($markup, "text/xml{$contentType}");
|
| 4659 |
+
}
|
| 4660 |
+
/**
|
| 4661 |
+
* Creates new document from markup.
|
| 4662 |
+
* Chainable.
|
| 4663 |
+
*
|
| 4664 |
+
* @param unknown_type $markup
|
| 4665 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4666 |
+
*/
|
| 4667 |
+
public static function newDocumentXHTML($markup = null, $charset = null) {
|
| 4668 |
+
$contentType = $charset
|
| 4669 |
+
? ";charset=$charset"
|
| 4670 |
+
: '';
|
| 4671 |
+
return self::newDocument($markup, "application/xhtml+xml{$contentType}");
|
| 4672 |
+
}
|
| 4673 |
+
/**
|
| 4674 |
+
* Creates new document from markup.
|
| 4675 |
+
* Chainable.
|
| 4676 |
+
*
|
| 4677 |
+
* @param unknown_type $markup
|
| 4678 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4679 |
+
*/
|
| 4680 |
+
public static function newDocumentPHP($markup = null, $contentType = "text/html") {
|
| 4681 |
+
// TODO pass charset to phpToMarkup if possible (use DOMDocumentWrapper function)
|
| 4682 |
+
$markup = phpQuery::phpToMarkup($markup, self::$defaultCharset);
|
| 4683 |
+
return self::newDocument($markup, $contentType);
|
| 4684 |
+
}
|
| 4685 |
+
public static function phpToMarkup($php, $charset = 'utf-8') {
|
| 4686 |
+
$regexes = array(
|
| 4687 |
+
'@(<(?!\\?)(?:[^>]|\\?>)+\\w+\\s*=\\s*)(\')([^\']*)<'.'?php?(.*?)(?:\\?>)([^\']*)\'@s',
|
| 4688 |
+
'@(<(?!\\?)(?:[^>]|\\?>)+\\w+\\s*=\\s*)(")([^"]*)<'.'?php?(.*?)(?:\\?>)([^"]*)"@s',
|
| 4689 |
+
);
|
| 4690 |
+
foreach($regexes as $regex)
|
| 4691 |
+
while (preg_match($regex, $php, $matches)) {
|
| 4692 |
+
$php = preg_replace_callback(
|
| 4693 |
+
$regex,
|
| 4694 |
+
// create_function('$m, $charset = "'.$charset.'"',
|
| 4695 |
+
// 'return $m[1].$m[2]
|
| 4696 |
+
// .htmlspecialchars("<"."?php".$m[4]."?".">", ENT_QUOTES|ENT_NOQUOTES, $charset)
|
| 4697 |
+
// .$m[5].$m[2];'
|
| 4698 |
+
// ),
|
| 4699 |
+
array('phpQuery', '_phpToMarkupCallback'),
|
| 4700 |
+
$php
|
| 4701 |
+
);
|
| 4702 |
+
}
|
| 4703 |
+
$regex = '@(^|>[^<]*)+?(<\?php(.*?)(\?>))@s';
|
| 4704 |
+
//preg_match_all($regex, $php, $matches);
|
| 4705 |
+
//var_dump($matches);
|
| 4706 |
+
$php = preg_replace($regex, '\\1<php><!-- \\3 --></php>', $php);
|
| 4707 |
+
return $php;
|
| 4708 |
+
}
|
| 4709 |
+
public static function _phpToMarkupCallback($php, $charset = 'utf-8') {
|
| 4710 |
+
return $m[1].$m[2]
|
| 4711 |
+
.htmlspecialchars("<"."?php".$m[4]."?".">", ENT_QUOTES|ENT_NOQUOTES, $charset)
|
| 4712 |
+
.$m[5].$m[2];
|
| 4713 |
+
}
|
| 4714 |
+
public static function _markupToPHPCallback($m) {
|
| 4715 |
+
return "<"."?php ".htmlspecialchars_decode($m[1])." ?".">";
|
| 4716 |
+
}
|
| 4717 |
+
/**
|
| 4718 |
+
* Converts document markup containing PHP code generated by phpQuery::php()
|
| 4719 |
+
* into valid (executable) PHP code syntax.
|
| 4720 |
+
*
|
| 4721 |
+
* @param string|phpQueryObject $content
|
| 4722 |
+
* @return string PHP code.
|
| 4723 |
+
*/
|
| 4724 |
+
public static function markupToPHP($content) {
|
| 4725 |
+
if ($content instanceof phpQueryObject)
|
| 4726 |
+
$content = $content->markupOuter();
|
| 4727 |
+
/* <php>...</php> to <?php...? > */
|
| 4728 |
+
$content = preg_replace_callback(
|
| 4729 |
+
'@<php>\s*<!--(.*?)-->\s*</php>@s',
|
| 4730 |
+
// create_function('$m',
|
| 4731 |
+
// 'return "<'.'?php ".htmlspecialchars_decode($m[1])." ?'.'>";'
|
| 4732 |
+
// ),
|
| 4733 |
+
array('phpQuery', '_markupToPHPCallback'),
|
| 4734 |
+
$content
|
| 4735 |
+
);
|
| 4736 |
+
/* <node attr='< ?php ? >'> extra space added to save highlighters */
|
| 4737 |
+
$regexes = array(
|
| 4738 |
+
'@(<(?!\\?)(?:[^>]|\\?>)+\\w+\\s*=\\s*)(\')([^\']*)(?:<|%3C)\\?(?:php)?(.*?)(?:\\?(?:>|%3E))([^\']*)\'@s',
|
| 4739 |
+
'@(<(?!\\?)(?:[^>]|\\?>)+\\w+\\s*=\\s*)(")([^"]*)(?:<|%3C)\\?(?:php)?(.*?)(?:\\?(?:>|%3E))([^"]*)"@s',
|
| 4740 |
+
);
|
| 4741 |
+
foreach($regexes as $regex)
|
| 4742 |
+
while (preg_match($regex, $content))
|
| 4743 |
+
$content = preg_replace_callback(
|
| 4744 |
+
$regex,
|
| 4745 |
+
create_function('$m',
|
| 4746 |
+
'return $m[1].$m[2].$m[3]."<?php "
|
| 4747 |
+
.str_replace(
|
| 4748 |
+
array("%20", "%3E", "%09", " ", "	", "%7B", "%24", "%7D", "%22", "%5B", "%5D"),
|
| 4749 |
+
array(" ", ">", " ", "\n", " ", "{", "$", "}", \'"\', "[", "]"),
|
| 4750 |
+
htmlspecialchars_decode($m[4])
|
| 4751 |
+
)
|
| 4752 |
+
." ?>".$m[5].$m[2];'
|
| 4753 |
+
),
|
| 4754 |
+
$content
|
| 4755 |
+
);
|
| 4756 |
+
return $content;
|
| 4757 |
+
}
|
| 4758 |
+
/**
|
| 4759 |
+
* Creates new document from file $file.
|
| 4760 |
+
* Chainable.
|
| 4761 |
+
*
|
| 4762 |
+
* @param string $file URLs allowed. See File wrapper page at php.net for more supported sources.
|
| 4763 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4764 |
+
*/
|
| 4765 |
+
public static function newDocumentFile($file, $contentType = null) {
|
| 4766 |
+
$documentID = self::createDocumentWrapper(
|
| 4767 |
+
file_get_contents($file), $contentType
|
| 4768 |
+
);
|
| 4769 |
+
return new phpQueryObject($documentID);
|
| 4770 |
+
}
|
| 4771 |
+
/**
|
| 4772 |
+
* Creates new document from markup.
|
| 4773 |
+
* Chainable.
|
| 4774 |
+
*
|
| 4775 |
+
* @param unknown_type $markup
|
| 4776 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4777 |
+
*/
|
| 4778 |
+
public static function newDocumentFileHTML($file, $charset = null) {
|
| 4779 |
+
$contentType = $charset
|
| 4780 |
+
? ";charset=$charset"
|
| 4781 |
+
: '';
|
| 4782 |
+
return self::newDocumentFile($file, "text/html{$contentType}");
|
| 4783 |
+
}
|
| 4784 |
+
/**
|
| 4785 |
+
* Creates new document from markup.
|
| 4786 |
+
* Chainable.
|
| 4787 |
+
*
|
| 4788 |
+
* @param unknown_type $markup
|
| 4789 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4790 |
+
*/
|
| 4791 |
+
public static function newDocumentFileXML($file, $charset = null) {
|
| 4792 |
+
$contentType = $charset
|
| 4793 |
+
? ";charset=$charset"
|
| 4794 |
+
: '';
|
| 4795 |
+
return self::newDocumentFile($file, "text/xml{$contentType}");
|
| 4796 |
+
}
|
| 4797 |
+
/**
|
| 4798 |
+
* Creates new document from markup.
|
| 4799 |
+
* Chainable.
|
| 4800 |
+
*
|
| 4801 |
+
* @param unknown_type $markup
|
| 4802 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4803 |
+
*/
|
| 4804 |
+
public static function newDocumentFileXHTML($file, $charset = null) {
|
| 4805 |
+
$contentType = $charset
|
| 4806 |
+
? ";charset=$charset"
|
| 4807 |
+
: '';
|
| 4808 |
+
return self::newDocumentFile($file, "application/xhtml+xml{$contentType}");
|
| 4809 |
+
}
|
| 4810 |
+
/**
|
| 4811 |
+
* Creates new document from markup.
|
| 4812 |
+
* Chainable.
|
| 4813 |
+
*
|
| 4814 |
+
* @param unknown_type $markup
|
| 4815 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4816 |
+
*/
|
| 4817 |
+
public static function newDocumentFilePHP($file, $contentType = null) {
|
| 4818 |
+
return self::newDocumentPHP(file_get_contents($file), $contentType);
|
| 4819 |
+
}
|
| 4820 |
+
/**
|
| 4821 |
+
* Reuses existing DOMDocument object.
|
| 4822 |
+
* Chainable.
|
| 4823 |
+
*
|
| 4824 |
+
* @param $document DOMDocument
|
| 4825 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 4826 |
+
* @TODO support DOMDocument
|
| 4827 |
+
*/
|
| 4828 |
+
public static function loadDocument($document) {
|
| 4829 |
+
// TODO
|
| 4830 |
+
die('TODO loadDocument');
|
| 4831 |
+
}
|
| 4832 |
+
/**
|
| 4833 |
+
* Enter description here...
|
| 4834 |
+
*
|
| 4835 |
+
* @param unknown_type $html
|
| 4836 |
+
* @param unknown_type $domId
|
| 4837 |
+
* @return unknown New DOM ID
|
| 4838 |
+
* @todo support PHP tags in input
|
| 4839 |
+
* @todo support passing DOMDocument object from self::loadDocument
|
| 4840 |
+
*/
|
| 4841 |
+
protected static function createDocumentWrapper($html, $contentType = null, $documentID = null) {
|
| 4842 |
+
if (function_exists('domxml_open_mem'))
|
| 4843 |
+
throw new Exception("Old PHP4 DOM XML extension detected. phpQuery won't work until this extension is enabled.");
|
| 4844 |
+
// $id = $documentID
|
| 4845 |
+
// ? $documentID
|
| 4846 |
+
// : md5(microtime());
|
| 4847 |
+
$document = null;
|
| 4848 |
+
if ($html instanceof DOMDOCUMENT) {
|
| 4849 |
+
if (self::getDocumentID($html)) {
|
| 4850 |
+
// document already exists in phpQuery::$documents, make a copy
|
| 4851 |
+
$document = clone $html;
|
| 4852 |
+
} else {
|
| 4853 |
+
// new document, add it to phpQuery::$documents
|
| 4854 |
+
$wrapper = new DOMDocumentWrapper($html, $contentType, $documentID);
|
| 4855 |
+
}
|
| 4856 |
+
} else {
|
| 4857 |
+
$wrapper = new DOMDocumentWrapper($html, $contentType, $documentID);
|
| 4858 |
+
}
|
| 4859 |
+
// $wrapper->id = $id;
|
| 4860 |
+
// bind document
|
| 4861 |
+
phpQuery::$documents[$wrapper->id] = $wrapper;
|
| 4862 |
+
// remember last loaded document
|
| 4863 |
+
phpQuery::selectDocument($wrapper->id);
|
| 4864 |
+
return $wrapper->id;
|
| 4865 |
+
}
|
| 4866 |
+
/**
|
| 4867 |
+
* Extend class namespace.
|
| 4868 |
+
*
|
| 4869 |
+
* @param string|array $target
|
| 4870 |
+
* @param array $source
|
| 4871 |
+
* @TODO support string $source
|
| 4872 |
+
* @return unknown_type
|
| 4873 |
+
*/
|
| 4874 |
+
public static function extend($target, $source) {
|
| 4875 |
+
switch($target) {
|
| 4876 |
+
case 'phpQueryObject':
|
| 4877 |
+
$targetRef = &self::$extendMethods;
|
| 4878 |
+
$targetRef2 = &self::$pluginsMethods;
|
| 4879 |
+
break;
|
| 4880 |
+
case 'phpQuery':
|
| 4881 |
+
$targetRef = &self::$extendStaticMethods;
|
| 4882 |
+
$targetRef2 = &self::$pluginsStaticMethods;
|
| 4883 |
+
break;
|
| 4884 |
+
default:
|
| 4885 |
+
throw new Exception("Unsupported \$target type");
|
| 4886 |
+
}
|
| 4887 |
+
if (is_string($source))
|
| 4888 |
+
$source = array($source => $source);
|
| 4889 |
+
foreach($source as $method => $callback) {
|
| 4890 |
+
if (isset($targetRef[$method])) {
|
| 4891 |
+
// throw new Exception
|
| 4892 |
+
self::debug("Duplicate method '{$method}', can\'t extend '{$target}'");
|
| 4893 |
+
continue;
|
| 4894 |
+
}
|
| 4895 |
+
if (isset($targetRef2[$method])) {
|
| 4896 |
+
// throw new Exception
|
| 4897 |
+
self::debug("Duplicate method '{$method}' from plugin '{$targetRef2[$method]}',"
|
| 4898 |
+
." can\'t extend '{$target}'");
|
| 4899 |
+
continue;
|
| 4900 |
+
}
|
| 4901 |
+
$targetRef[$method] = $callback;
|
| 4902 |
+
}
|
| 4903 |
+
return true;
|
| 4904 |
+
}
|
| 4905 |
+
/**
|
| 4906 |
+
* Extend phpQuery with $class from $file.
|
| 4907 |
+
*
|
| 4908 |
+
* @param string $class Extending class name. Real class name can be prepended phpQuery_.
|
| 4909 |
+
* @param string $file Filename to include. Defaults to "{$class}.php".
|
| 4910 |
+
*/
|
| 4911 |
+
public static function plugin($class, $file = null) {
|
| 4912 |
+
// TODO $class checked agains phpQuery_$class
|
| 4913 |
+
// if (strpos($class, 'phpQuery') === 0)
|
| 4914 |
+
// $class = substr($class, 8);
|
| 4915 |
+
if (in_array($class, self::$pluginsLoaded))
|
| 4916 |
+
return true;
|
| 4917 |
+
if (! $file)
|
| 4918 |
+
$file = $class.'.php';
|
| 4919 |
+
$objectClassExists = class_exists('phpQueryObjectPlugin_'.$class);
|
| 4920 |
+
$staticClassExists = class_exists('phpQueryPlugin_'.$class);
|
| 4921 |
+
if (! $objectClassExists && ! $staticClassExists)
|
| 4922 |
+
require_once($file);
|
| 4923 |
+
self::$pluginsLoaded[] = $class;
|
| 4924 |
+
// static methods
|
| 4925 |
+
if (class_exists('phpQueryPlugin_'.$class)) {
|
| 4926 |
+
$realClass = 'phpQueryPlugin_'.$class;
|
| 4927 |
+
$vars = get_class_vars($realClass);
|
| 4928 |
+
$loop = isset($vars['phpQueryMethods'])
|
| 4929 |
+
&& ! is_null($vars['phpQueryMethods'])
|
| 4930 |
+
? $vars['phpQueryMethods']
|
| 4931 |
+
: get_class_methods($realClass);
|
| 4932 |
+
foreach($loop as $method) {
|
| 4933 |
+
if ($method == '__initialize')
|
| 4934 |
+
continue;
|
| 4935 |
+
if (! is_callable(array($realClass, $method)))
|
| 4936 |
+
continue;
|
| 4937 |
+
if (isset(self::$pluginsStaticMethods[$method])) {
|
| 4938 |
+
throw new Exception("Duplicate method '{$method}' from plugin '{$c}' conflicts with same method from plugin '".self::$pluginsStaticMethods[$method]."'");
|
| 4939 |
+
return;
|
| 4940 |
+
}
|
| 4941 |
+
self::$pluginsStaticMethods[$method] = $class;
|
| 4942 |
+
}
|
| 4943 |
+
if (method_exists($realClass, '__initialize'))
|
| 4944 |
+
call_user_func_array(array($realClass, '__initialize'), array());
|
| 4945 |
+
}
|
| 4946 |
+
// object methods
|
| 4947 |
+
if (class_exists('phpQueryObjectPlugin_'.$class)) {
|
| 4948 |
+
$realClass = 'phpQueryObjectPlugin_'.$class;
|
| 4949 |
+
$vars = get_class_vars($realClass);
|
| 4950 |
+
$loop = isset($vars['phpQueryMethods'])
|
| 4951 |
+
&& ! is_null($vars['phpQueryMethods'])
|
| 4952 |
+
? $vars['phpQueryMethods']
|
| 4953 |
+
: get_class_methods($realClass);
|
| 4954 |
+
foreach($loop as $method) {
|
| 4955 |
+
if (! is_callable(array($realClass, $method)))
|
| 4956 |
+
continue;
|
| 4957 |
+
if (isset(self::$pluginsMethods[$method])) {
|
| 4958 |
+
throw new Exception("Duplicate method '{$method}' from plugin '{$c}' conflicts with same method from plugin '".self::$pluginsMethods[$method]."'");
|
| 4959 |
+
continue;
|
| 4960 |
+
}
|
| 4961 |
+
self::$pluginsMethods[$method] = $class;
|
| 4962 |
+
}
|
| 4963 |
+
}
|
| 4964 |
+
return true;
|
| 4965 |
+
}
|
| 4966 |
+
/**
|
| 4967 |
+
* Unloades all or specified document from memory.
|
| 4968 |
+
*
|
| 4969 |
+
* @param mixed $documentID @see phpQuery::getDocumentID() for supported types.
|
| 4970 |
+
*/
|
| 4971 |
+
public static function unloadDocuments($id = null) {
|
| 4972 |
+
if (isset($id)) {
|
| 4973 |
+
if ($id = self::getDocumentID($id))
|
| 4974 |
+
unset(phpQuery::$documents[$id]);
|
| 4975 |
+
} else {
|
| 4976 |
+
foreach(phpQuery::$documents as $k => $v) {
|
| 4977 |
+
unset(phpQuery::$documents[$k]);
|
| 4978 |
+
}
|
| 4979 |
+
}
|
| 4980 |
+
}
|
| 4981 |
+
/**
|
| 4982 |
+
* Parses phpQuery object or HTML result against PHP tags and makes them active.
|
| 4983 |
+
*
|
| 4984 |
+
* @param phpQuery|string $content
|
| 4985 |
+
* @deprecated
|
| 4986 |
+
* @return string
|
| 4987 |
+
*/
|
| 4988 |
+
public static function unsafePHPTags($content) {
|
| 4989 |
+
return self::markupToPHP($content);
|
| 4990 |
+
}
|
| 4991 |
+
public static function DOMNodeListToArray($DOMNodeList) {
|
| 4992 |
+
$array = array();
|
| 4993 |
+
if (! $DOMNodeList)
|
| 4994 |
+
return $array;
|
| 4995 |
+
foreach($DOMNodeList as $node)
|
| 4996 |
+
$array[] = $node;
|
| 4997 |
+
return $array;
|
| 4998 |
+
}
|
| 4999 |
+
/**
|
| 5000 |
+
* Checks if $input is HTML string, which has to start with '<'.
|
| 5001 |
+
*
|
| 5002 |
+
* @deprecated
|
| 5003 |
+
* @param String $input
|
| 5004 |
+
* @return Bool
|
| 5005 |
+
* @todo still used ?
|
| 5006 |
+
*/
|
| 5007 |
+
public static function isMarkup($input) {
|
| 5008 |
+
return ! is_array($input) && substr(trim($input), 0, 1) == '<';
|
| 5009 |
+
}
|
| 5010 |
+
public static function debug($text) {
|
| 5011 |
+
if (self::$debug)
|
| 5012 |
+
print var_dump($text);
|
| 5013 |
+
}
|
| 5014 |
+
/**
|
| 5015 |
+
* Make an AJAX request.
|
| 5016 |
+
*
|
| 5017 |
+
* @param array See $options http://docs.jquery.com/Ajax/jQuery.ajax#toptions
|
| 5018 |
+
* Additional options are:
|
| 5019 |
+
* 'document' - document for global events, @see phpQuery::getDocumentID()
|
| 5020 |
+
* 'referer' - implemented
|
| 5021 |
+
* 'requested_with' - TODO; not implemented (X-Requested-With)
|
| 5022 |
+
* @return Zend_Http_Client
|
| 5023 |
+
* @link http://docs.jquery.com/Ajax/jQuery.ajax
|
| 5024 |
+
*
|
| 5025 |
+
* @TODO $options['cache']
|
| 5026 |
+
* @TODO $options['processData']
|
| 5027 |
+
* @TODO $options['xhr']
|
| 5028 |
+
* @TODO $options['data'] as string
|
| 5029 |
+
* @TODO XHR interface
|
| 5030 |
+
*/
|
| 5031 |
+
public static function ajax($options = array(), $xhr = null) {
|
| 5032 |
+
$options = array_merge(
|
| 5033 |
+
self::$ajaxSettings, $options
|
| 5034 |
+
);
|
| 5035 |
+
$documentID = isset($options['document'])
|
| 5036 |
+
? self::getDocumentID($options['document'])
|
| 5037 |
+
: null;
|
| 5038 |
+
if ($xhr) {
|
| 5039 |
+
// reuse existing XHR object, but clean it up
|
| 5040 |
+
$client = $xhr;
|
| 5041 |
+
// $client->setParameterPost(null);
|
| 5042 |
+
// $client->setParameterGet(null);
|
| 5043 |
+
$client->setAuth(false);
|
| 5044 |
+
$client->setHeaders("If-Modified-Since", null);
|
| 5045 |
+
$client->setHeaders("Referer", null);
|
| 5046 |
+
$client->resetParameters();
|
| 5047 |
+
} else {
|
| 5048 |
+
// create new XHR object
|
| 5049 |
+
require_once('Zend/Http/Client.php');
|
| 5050 |
+
$client = new Zend_Http_Client();
|
| 5051 |
+
$client->setCookieJar();
|
| 5052 |
+
}
|
| 5053 |
+
if (isset($options['timeout']))
|
| 5054 |
+
$client->setConfig(array(
|
| 5055 |
+
'timeout' => $options['timeout'],
|
| 5056 |
+
));
|
| 5057 |
+
// 'maxredirects' => 0,
|
| 5058 |
+
foreach(self::$ajaxAllowedHosts as $k => $host)
|
| 5059 |
+
if ($host == '.' && isset($_SERVER['HTTP_HOST']))
|
| 5060 |
+
self::$ajaxAllowedHosts[$k] = $_SERVER['HTTP_HOST'];
|
| 5061 |
+
$host = parse_url($options['url'], PHP_URL_HOST);
|
| 5062 |
+
if (! in_array($host, self::$ajaxAllowedHosts)) {
|
| 5063 |
+
throw new Exception("Request not permitted, host '$host' not present in "
|
| 5064 |
+
."phpQuery::\$ajaxAllowedHosts");
|
| 5065 |
+
}
|
| 5066 |
+
// JSONP
|
| 5067 |
+
$jsre = "/=\\?(&|$)/";
|
| 5068 |
+
if (isset($options['dataType']) && $options['dataType'] == 'jsonp') {
|
| 5069 |
+
$jsonpCallbackParam = $options['jsonp']
|
| 5070 |
+
? $options['jsonp'] : 'callback';
|
| 5071 |
+
if (strtolower($options['type']) == 'get') {
|
| 5072 |
+
if (! preg_match($jsre, $options['url'])) {
|
| 5073 |
+
$sep = strpos($options['url'], '?')
|
| 5074 |
+
? '&' : '?';
|
| 5075 |
+
$options['url'] .= "$sep$jsonpCallbackParam=?";
|
| 5076 |
+
}
|
| 5077 |
+
} else if ($options['data']) {
|
| 5078 |
+
$jsonp = false;
|
| 5079 |
+
foreach($options['data'] as $n => $v) {
|
| 5080 |
+
if ($v == '?')
|
| 5081 |
+
$jsonp = true;
|
| 5082 |
+
}
|
| 5083 |
+
if (! $jsonp) {
|
| 5084 |
+
$options['data'][$jsonpCallbackParam] = '?';
|
| 5085 |
+
}
|
| 5086 |
+
}
|
| 5087 |
+
$options['dataType'] = 'json';
|
| 5088 |
+
}
|
| 5089 |
+
if (isset($options['dataType']) && $options['dataType'] == 'json') {
|
| 5090 |
+
$jsonpCallback = 'json_'.md5(microtime());
|
| 5091 |
+
$jsonpData = $jsonpUrl = false;
|
| 5092 |
+
if ($options['data']) {
|
| 5093 |
+
foreach($options['data'] as $n => $v) {
|
| 5094 |
+
if ($v == '?')
|
| 5095 |
+
$jsonpData = $n;
|
| 5096 |
+
}
|
| 5097 |
+
}
|
| 5098 |
+
if (preg_match($jsre, $options['url']))
|
| 5099 |
+
$jsonpUrl = true;
|
| 5100 |
+
if ($jsonpData !== false || $jsonpUrl) {
|
| 5101 |
+
// remember callback name for httpData()
|
| 5102 |
+
$options['_jsonp'] = $jsonpCallback;
|
| 5103 |
+
if ($jsonpData !== false)
|
| 5104 |
+
$options['data'][$jsonpData] = $jsonpCallback;
|
| 5105 |
+
if ($jsonpUrl)
|
| 5106 |
+
$options['url'] = preg_replace($jsre, "=$jsonpCallback\\1", $options['url']);
|
| 5107 |
+
}
|
| 5108 |
+
}
|
| 5109 |
+
$client->setUri($options['url']);
|
| 5110 |
+
$client->setMethod(strtoupper($options['type']));
|
| 5111 |
+
if (isset($options['referer']) && $options['referer'])
|
| 5112 |
+
$client->setHeaders('Referer', $options['referer']);
|
| 5113 |
+
$client->setHeaders(array(
|
| 5114 |
+
// 'content-type' => $options['contentType'],
|
| 5115 |
+
'User-Agent' => 'Mozilla/5.0 (X11; U; Linux x86; en-US; rv:1.9.0.5) Gecko'
|
| 5116 |
+
.'/2008122010 Firefox/3.0.5',
|
| 5117 |
+
// TODO custom charset
|
| 5118 |
+
'Accept-Charset' => 'ISO-8859-1,utf-8;q=0.7,*;q=0.7',
|
| 5119 |
+
// 'Connection' => 'keep-alive',
|
| 5120 |
+
// 'Accept' => 'text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8',
|
| 5121 |
+
'Accept-Language' => 'en-us,en;q=0.5',
|
| 5122 |
+
));
|
| 5123 |
+
if ($options['username'])
|
| 5124 |
+
$client->setAuth($options['username'], $options['password']);
|
| 5125 |
+
if (isset($options['ifModified']) && $options['ifModified'])
|
| 5126 |
+
$client->setHeaders("If-Modified-Since",
|
| 5127 |
+
self::$lastModified
|
| 5128 |
+
? self::$lastModified
|
| 5129 |
+
: "Thu, 01 Jan 1970 00:00:00 GMT"
|
| 5130 |
+
);
|
| 5131 |
+
$client->setHeaders("Accept",
|
| 5132 |
+
isset($options['dataType'])
|
| 5133 |
+
&& isset(self::$ajaxSettings['accepts'][ $options['dataType'] ])
|
| 5134 |
+
? self::$ajaxSettings['accepts'][ $options['dataType'] ].", */*"
|
| 5135 |
+
: self::$ajaxSettings['accepts']['_default']
|
| 5136 |
+
);
|
| 5137 |
+
// TODO $options['processData']
|
| 5138 |
+
if ($options['data'] instanceof phpQueryObject) {
|
| 5139 |
+
$serialized = $options['data']->serializeArray($options['data']);
|
| 5140 |
+
$options['data'] = array();
|
| 5141 |
+
foreach($serialized as $r)
|
| 5142 |
+
$options['data'][ $r['name'] ] = $r['value'];
|
| 5143 |
+
}
|
| 5144 |
+
if (strtolower($options['type']) == 'get') {
|
| 5145 |
+
$client->setParameterGet($options['data']);
|
| 5146 |
+
} else if (strtolower($options['type']) == 'post') {
|
| 5147 |
+
$client->setEncType($options['contentType']);
|
| 5148 |
+
$client->setParameterPost($options['data']);
|
| 5149 |
+
}
|
| 5150 |
+
if (self::$active == 0 && $options['global'])
|
| 5151 |
+
phpQueryEvents::trigger($documentID, 'ajaxStart');
|
| 5152 |
+
self::$active++;
|
| 5153 |
+
// beforeSend callback
|
| 5154 |
+
if (isset($options['beforeSend']) && $options['beforeSend'])
|
| 5155 |
+
phpQuery::callbackRun($options['beforeSend'], array($client));
|
| 5156 |
+
// ajaxSend event
|
| 5157 |
+
if ($options['global'])
|
| 5158 |
+
phpQueryEvents::trigger($documentID, 'ajaxSend', array($client, $options));
|
| 5159 |
+
if (phpQuery::$debug) {
|
| 5160 |
+
self::debug("{$options['type']}: {$options['url']}\n");
|
| 5161 |
+
self::debug("Options: <pre>".var_export($options, true)."</pre>\n");
|
| 5162 |
+
// if ($client->getCookieJar())
|
| 5163 |
+
// self::debug("Cookies: <pre>".var_export($client->getCookieJar()->getMatchingCookies($options['url']), true)."</pre>\n");
|
| 5164 |
+
}
|
| 5165 |
+
// request
|
| 5166 |
+
$response = $client->request();
|
| 5167 |
+
if (phpQuery::$debug) {
|
| 5168 |
+
self::debug('Status: '.$response->getStatus().' / '.$response->getMessage());
|
| 5169 |
+
self::debug($client->getLastRequest());
|
| 5170 |
+
self::debug($response->getHeaders());
|
| 5171 |
+
}
|
| 5172 |
+
if ($response->isSuccessful()) {
|
| 5173 |
+
// XXX tempolary
|
| 5174 |
+
self::$lastModified = $response->getHeader('Last-Modified');
|
| 5175 |
+
$data = self::httpData($response->getBody(), $options['dataType'], $options);
|
| 5176 |
+
if (isset($options['success']) && $options['success'])
|
| 5177 |
+
phpQuery::callbackRun($options['success'], array($data, $response->getStatus(), $options));
|
| 5178 |
+
if ($options['global'])
|
| 5179 |
+
phpQueryEvents::trigger($documentID, 'ajaxSuccess', array($client, $options));
|
| 5180 |
+
} else {
|
| 5181 |
+
if (isset($options['error']) && $options['error'])
|
| 5182 |
+
phpQuery::callbackRun($options['error'], array($client, $response->getStatus(), $response->getMessage()));
|
| 5183 |
+
if ($options['global'])
|
| 5184 |
+
phpQueryEvents::trigger($documentID, 'ajaxError', array($client, /*$response->getStatus(),*/$response->getMessage(), $options));
|
| 5185 |
+
}
|
| 5186 |
+
if (isset($options['complete']) && $options['complete'])
|
| 5187 |
+
phpQuery::callbackRun($options['complete'], array($client, $response->getStatus()));
|
| 5188 |
+
if ($options['global'])
|
| 5189 |
+
phpQueryEvents::trigger($documentID, 'ajaxComplete', array($client, $options));
|
| 5190 |
+
if ($options['global'] && ! --self::$active)
|
| 5191 |
+
phpQueryEvents::trigger($documentID, 'ajaxStop');
|
| 5192 |
+
return $client;
|
| 5193 |
+
// if (is_null($domId))
|
| 5194 |
+
// $domId = self::$defaultDocumentID ? self::$defaultDocumentID : false;
|
| 5195 |
+
// return new phpQueryAjaxResponse($response, $domId);
|
| 5196 |
+
}
|
| 5197 |
+
protected static function httpData($data, $type, $options) {
|
| 5198 |
+
if (isset($options['dataFilter']) && $options['dataFilter'])
|
| 5199 |
+
$data = self::callbackRun($options['dataFilter'], array($data, $type));
|
| 5200 |
+
if (is_string($data)) {
|
| 5201 |
+
if ($type == "json") {
|
| 5202 |
+
if (isset($options['_jsonp']) && $options['_jsonp']) {
|
| 5203 |
+
$data = preg_replace('/^\s*\w+\((.*)\)\s*$/s', '$1', $data);
|
| 5204 |
+
}
|
| 5205 |
+
$data = self::parseJSON($data);
|
| 5206 |
+
}
|
| 5207 |
+
}
|
| 5208 |
+
return $data;
|
| 5209 |
+
}
|
| 5210 |
+
/**
|
| 5211 |
+
* Enter description here...
|
| 5212 |
+
*
|
| 5213 |
+
* @param array|phpQuery $data
|
| 5214 |
+
*
|
| 5215 |
+
*/
|
| 5216 |
+
public static function param($data) {
|
| 5217 |
+
return http_build_query($data, null, '&');
|
| 5218 |
+
}
|
| 5219 |
+
public static function get($url, $data = null, $callback = null, $type = null) {
|
| 5220 |
+
if (!is_array($data)) {
|
| 5221 |
+
$callback = $data;
|
| 5222 |
+
$data = null;
|
| 5223 |
+
}
|
| 5224 |
+
// TODO some array_values on this shit
|
| 5225 |
+
return phpQuery::ajax(array(
|
| 5226 |
+
'type' => 'GET',
|
| 5227 |
+
'url' => $url,
|
| 5228 |
+
'data' => $data,
|
| 5229 |
+
'success' => $callback,
|
| 5230 |
+
'dataType' => $type,
|
| 5231 |
+
));
|
| 5232 |
+
}
|
| 5233 |
+
public static function post($url, $data = null, $callback = null, $type = null) {
|
| 5234 |
+
if (!is_array($data)) {
|
| 5235 |
+
$callback = $data;
|
| 5236 |
+
$data = null;
|
| 5237 |
+
}
|
| 5238 |
+
return phpQuery::ajax(array(
|
| 5239 |
+
'type' => 'POST',
|
| 5240 |
+
'url' => $url,
|
| 5241 |
+
'data' => $data,
|
| 5242 |
+
'success' => $callback,
|
| 5243 |
+
'dataType' => $type,
|
| 5244 |
+
));
|
| 5245 |
+
}
|
| 5246 |
+
public static function getJSON($url, $data = null, $callback = null) {
|
| 5247 |
+
if (!is_array($data)) {
|
| 5248 |
+
$callback = $data;
|
| 5249 |
+
$data = null;
|
| 5250 |
+
}
|
| 5251 |
+
// TODO some array_values on this shit
|
| 5252 |
+
return phpQuery::ajax(array(
|
| 5253 |
+
'type' => 'GET',
|
| 5254 |
+
'url' => $url,
|
| 5255 |
+
'data' => $data,
|
| 5256 |
+
'success' => $callback,
|
| 5257 |
+
'dataType' => 'json',
|
| 5258 |
+
));
|
| 5259 |
+
}
|
| 5260 |
+
public static function ajaxSetup($options) {
|
| 5261 |
+
self::$ajaxSettings = array_merge(
|
| 5262 |
+
self::$ajaxSettings,
|
| 5263 |
+
$options
|
| 5264 |
+
);
|
| 5265 |
+
}
|
| 5266 |
+
public static function ajaxAllowHost($host1, $host2 = null, $host3 = null) {
|
| 5267 |
+
$loop = is_array($host1)
|
| 5268 |
+
? $host1
|
| 5269 |
+
: func_get_args();
|
| 5270 |
+
foreach($loop as $host) {
|
| 5271 |
+
if ($host && ! in_array($host, phpQuery::$ajaxAllowedHosts)) {
|
| 5272 |
+
phpQuery::$ajaxAllowedHosts[] = $host;
|
| 5273 |
+
}
|
| 5274 |
+
}
|
| 5275 |
+
}
|
| 5276 |
+
public static function ajaxAllowURL($url1, $url2 = null, $url3 = null) {
|
| 5277 |
+
$loop = is_array($url1)
|
| 5278 |
+
? $url1
|
| 5279 |
+
: func_get_args();
|
| 5280 |
+
foreach($loop as $url)
|
| 5281 |
+
phpQuery::ajaxAllowHost(parse_url($url, PHP_URL_HOST));
|
| 5282 |
+
}
|
| 5283 |
+
/**
|
| 5284 |
+
* Returns JSON representation of $data.
|
| 5285 |
+
*
|
| 5286 |
+
* @static
|
| 5287 |
+
* @param mixed $data
|
| 5288 |
+
* @return string
|
| 5289 |
+
*/
|
| 5290 |
+
public static function toJSON($data) {
|
| 5291 |
+
if (function_exists('json_encode'))
|
| 5292 |
+
return json_encode($data);
|
| 5293 |
+
require_once('Zend/Json/Encoder.php');
|
| 5294 |
+
return Zend_Json_Encoder::encode($data);
|
| 5295 |
+
}
|
| 5296 |
+
/**
|
| 5297 |
+
* Parses JSON into proper PHP type.
|
| 5298 |
+
*
|
| 5299 |
+
* @static
|
| 5300 |
+
* @param string $json
|
| 5301 |
+
* @return mixed
|
| 5302 |
+
*/
|
| 5303 |
+
public static function parseJSON($json) {
|
| 5304 |
+
if (function_exists('json_decode')) {
|
| 5305 |
+
$return = json_decode(trim($json), true);
|
| 5306 |
+
// json_decode and UTF8 issues
|
| 5307 |
+
if (isset($return))
|
| 5308 |
+
return $return;
|
| 5309 |
+
}
|
| 5310 |
+
require_once('Zend/Json/Decoder.php');
|
| 5311 |
+
return Zend_Json_Decoder::decode($json);
|
| 5312 |
+
}
|
| 5313 |
+
/**
|
| 5314 |
+
* Returns source's document ID.
|
| 5315 |
+
*
|
| 5316 |
+
* @param $source DOMNode|phpQueryObject
|
| 5317 |
+
* @return string
|
| 5318 |
+
*/
|
| 5319 |
+
public static function getDocumentID($source) {
|
| 5320 |
+
if ($source instanceof DOMDOCUMENT) {
|
| 5321 |
+
foreach(phpQuery::$documents as $id => $document) {
|
| 5322 |
+
if ($source->isSameNode($document->document))
|
| 5323 |
+
return $id;
|
| 5324 |
+
}
|
| 5325 |
+
} else if ($source instanceof DOMNODE) {
|
| 5326 |
+
foreach(phpQuery::$documents as $id => $document) {
|
| 5327 |
+
if ($source->ownerDocument->isSameNode($document->document))
|
| 5328 |
+
return $id;
|
| 5329 |
+
}
|
| 5330 |
+
} else if ($source instanceof phpQueryObject)
|
| 5331 |
+
return $source->getDocumentID();
|
| 5332 |
+
else if (is_string($source) && isset(phpQuery::$documents[$source]))
|
| 5333 |
+
return $source;
|
| 5334 |
+
}
|
| 5335 |
+
/**
|
| 5336 |
+
* Get DOMDocument object related to $source.
|
| 5337 |
+
* Returns null if such document doesn't exist.
|
| 5338 |
+
*
|
| 5339 |
+
* @param $source DOMNode|phpQueryObject|string
|
| 5340 |
+
* @return string
|
| 5341 |
+
*/
|
| 5342 |
+
public static function getDOMDocument($source) {
|
| 5343 |
+
if ($source instanceof DOMDOCUMENT)
|
| 5344 |
+
return $source;
|
| 5345 |
+
$source = self::getDocumentID($source);
|
| 5346 |
+
return $source
|
| 5347 |
+
? self::$documents[$id]['document']
|
| 5348 |
+
: null;
|
| 5349 |
+
}
|
| 5350 |
+
|
| 5351 |
+
// UTILITIES
|
| 5352 |
+
// http://docs.jquery.com/Utilities
|
| 5353 |
+
|
| 5354 |
+
/**
|
| 5355 |
+
*
|
| 5356 |
+
* @return unknown_type
|
| 5357 |
+
* @link http://docs.jquery.com/Utilities/jQuery.makeArray
|
| 5358 |
+
*/
|
| 5359 |
+
public static function makeArray($obj) {
|
| 5360 |
+
$array = array();
|
| 5361 |
+
if (is_object($object) && $object instanceof DOMNODELIST) {
|
| 5362 |
+
foreach($object as $value)
|
| 5363 |
+
$array[] = $value;
|
| 5364 |
+
} else if (is_object($object) && ! ($object instanceof Iterator)) {
|
| 5365 |
+
foreach(get_object_vars($object) as $name => $value)
|
| 5366 |
+
$array[0][$name] = $value;
|
| 5367 |
+
} else {
|
| 5368 |
+
foreach($object as $name => $value)
|
| 5369 |
+
$array[0][$name] = $value;
|
| 5370 |
+
}
|
| 5371 |
+
return $array;
|
| 5372 |
+
}
|
| 5373 |
+
public static function inArray($value, $array) {
|
| 5374 |
+
return in_array($value, $array);
|
| 5375 |
+
}
|
| 5376 |
+
/**
|
| 5377 |
+
*
|
| 5378 |
+
* @param $object
|
| 5379 |
+
* @param $callback
|
| 5380 |
+
* @return unknown_type
|
| 5381 |
+
* @link http://docs.jquery.com/Utilities/jQuery.each
|
| 5382 |
+
*/
|
| 5383 |
+
public static function each($object, $callback, $param1 = null, $param2 = null, $param3 = null) {
|
| 5384 |
+
$paramStructure = null;
|
| 5385 |
+
if (func_num_args() > 2) {
|
| 5386 |
+
$paramStructure = func_get_args();
|
| 5387 |
+
$paramStructure = array_slice($paramStructure, 2);
|
| 5388 |
+
}
|
| 5389 |
+
if (is_object($object) && ! ($object instanceof Iterator)) {
|
| 5390 |
+
foreach(get_object_vars($object) as $name => $value)
|
| 5391 |
+
phpQuery::callbackRun($callback, array($name, $value), $paramStructure);
|
| 5392 |
+
} else {
|
| 5393 |
+
foreach($object as $name => $value)
|
| 5394 |
+
phpQuery::callbackRun($callback, array($name, $value), $paramStructure);
|
| 5395 |
+
}
|
| 5396 |
+
}
|
| 5397 |
+
/**
|
| 5398 |
+
*
|
| 5399 |
+
* @link http://docs.jquery.com/Utilities/jQuery.map
|
| 5400 |
+
*/
|
| 5401 |
+
public static function map($array, $callback, $param1 = null, $param2 = null, $param3 = null) {
|
| 5402 |
+
$result = array();
|
| 5403 |
+
$paramStructure = null;
|
| 5404 |
+
if (func_num_args() > 2) {
|
| 5405 |
+
$paramStructure = func_get_args();
|
| 5406 |
+
$paramStructure = array_slice($paramStructure, 2);
|
| 5407 |
+
}
|
| 5408 |
+
foreach($array as $v) {
|
| 5409 |
+
$vv = phpQuery::callbackRun($callback, array($v), $paramStructure);
|
| 5410 |
+
// $callbackArgs = $args;
|
| 5411 |
+
// foreach($args as $i => $arg) {
|
| 5412 |
+
// $callbackArgs[$i] = $arg instanceof CallbackParam
|
| 5413 |
+
// ? $v
|
| 5414 |
+
// : $arg;
|
| 5415 |
+
// }
|
| 5416 |
+
// $vv = call_user_func_array($callback, $callbackArgs);
|
| 5417 |
+
if (is_array($vv)) {
|
| 5418 |
+
foreach($vv as $vvv)
|
| 5419 |
+
$result[] = $vvv;
|
| 5420 |
+
} else if ($vv !== null) {
|
| 5421 |
+
$result[] = $vv;
|
| 5422 |
+
}
|
| 5423 |
+
}
|
| 5424 |
+
return $result;
|
| 5425 |
+
}
|
| 5426 |
+
/**
|
| 5427 |
+
*
|
| 5428 |
+
* @param $callback Callback
|
| 5429 |
+
* @param $params
|
| 5430 |
+
* @param $paramStructure
|
| 5431 |
+
* @return unknown_type
|
| 5432 |
+
*/
|
| 5433 |
+
public static function callbackRun($callback, $params = array(), $paramStructure = null) {
|
| 5434 |
+
if (! $callback)
|
| 5435 |
+
return;
|
| 5436 |
+
if ($callback instanceof CallbackParameterToReference) {
|
| 5437 |
+
// TODO support ParamStructure to select which $param push to reference
|
| 5438 |
+
if (isset($params[0]))
|
| 5439 |
+
$callback->callback = $params[0];
|
| 5440 |
+
return true;
|
| 5441 |
+
}
|
| 5442 |
+
if ($callback instanceof Callback) {
|
| 5443 |
+
$paramStructure = $callback->params;
|
| 5444 |
+
$callback = $callback->callback;
|
| 5445 |
+
}
|
| 5446 |
+
if (! $paramStructure)
|
| 5447 |
+
return call_user_func_array($callback, $params);
|
| 5448 |
+
$p = 0;
|
| 5449 |
+
foreach($paramStructure as $i => $v) {
|
| 5450 |
+
$paramStructure[$i] = $v instanceof CallbackParam
|
| 5451 |
+
? $params[$p++]
|
| 5452 |
+
: $v;
|
| 5453 |
+
}
|
| 5454 |
+
return call_user_func_array($callback, $paramStructure);
|
| 5455 |
+
}
|
| 5456 |
+
/**
|
| 5457 |
+
* Merge 2 phpQuery objects.
|
| 5458 |
+
* @param array $one
|
| 5459 |
+
* @param array $two
|
| 5460 |
+
* @protected
|
| 5461 |
+
* @todo node lists, phpQueryObject
|
| 5462 |
+
*/
|
| 5463 |
+
public static function merge($one, $two) {
|
| 5464 |
+
$elements = $one->elements;
|
| 5465 |
+
foreach($two->elements as $node) {
|
| 5466 |
+
$exists = false;
|
| 5467 |
+
foreach($elements as $node2) {
|
| 5468 |
+
if ($node2->isSameNode($node))
|
| 5469 |
+
$exists = true;
|
| 5470 |
+
}
|
| 5471 |
+
if (! $exists)
|
| 5472 |
+
$elements[] = $node;
|
| 5473 |
+
}
|
| 5474 |
+
return $elements;
|
| 5475 |
+
// $one = $one->newInstance();
|
| 5476 |
+
// $one->elements = $elements;
|
| 5477 |
+
// return $one;
|
| 5478 |
+
}
|
| 5479 |
+
/**
|
| 5480 |
+
*
|
| 5481 |
+
* @param $array
|
| 5482 |
+
* @param $callback
|
| 5483 |
+
* @param $invert
|
| 5484 |
+
* @return unknown_type
|
| 5485 |
+
* @link http://docs.jquery.com/Utilities/jQuery.grep
|
| 5486 |
+
*/
|
| 5487 |
+
public static function grep($array, $callback, $invert = false) {
|
| 5488 |
+
$result = array();
|
| 5489 |
+
foreach($array as $k => $v) {
|
| 5490 |
+
$r = call_user_func_array($callback, array($v, $k));
|
| 5491 |
+
if ($r === !(bool)$invert)
|
| 5492 |
+
$result[] = $v;
|
| 5493 |
+
}
|
| 5494 |
+
return $result;
|
| 5495 |
+
}
|
| 5496 |
+
public static function unique($array) {
|
| 5497 |
+
return array_unique($array);
|
| 5498 |
+
}
|
| 5499 |
+
/**
|
| 5500 |
+
*
|
| 5501 |
+
* @param $function
|
| 5502 |
+
* @return unknown_type
|
| 5503 |
+
* @TODO there are problems with non-static methods, second parameter pass it
|
| 5504 |
+
* but doesnt verify is method is really callable
|
| 5505 |
+
*/
|
| 5506 |
+
public static function isFunction($function) {
|
| 5507 |
+
return is_callable($function);
|
| 5508 |
+
}
|
| 5509 |
+
public static function trim($str) {
|
| 5510 |
+
return trim($str);
|
| 5511 |
+
}
|
| 5512 |
+
/* PLUGINS NAMESPACE */
|
| 5513 |
+
/**
|
| 5514 |
+
*
|
| 5515 |
+
* @param $url
|
| 5516 |
+
* @param $callback
|
| 5517 |
+
* @param $param1
|
| 5518 |
+
* @param $param2
|
| 5519 |
+
* @param $param3
|
| 5520 |
+
* @return phpQueryObject
|
| 5521 |
+
*/
|
| 5522 |
+
public static function browserGet($url, $callback, $param1 = null, $param2 = null, $param3 = null) {
|
| 5523 |
+
if (self::plugin('WebBrowser')) {
|
| 5524 |
+
$params = func_get_args();
|
| 5525 |
+
return self::callbackRun(array(self::$plugins, 'browserGet'), $params);
|
| 5526 |
+
} else {
|
| 5527 |
+
self::debug('WebBrowser plugin not available...');
|
| 5528 |
+
}
|
| 5529 |
+
}
|
| 5530 |
+
/**
|
| 5531 |
+
*
|
| 5532 |
+
* @param $url
|
| 5533 |
+
* @param $data
|
| 5534 |
+
* @param $callback
|
| 5535 |
+
* @param $param1
|
| 5536 |
+
* @param $param2
|
| 5537 |
+
* @param $param3
|
| 5538 |
+
* @return phpQueryObject
|
| 5539 |
+
*/
|
| 5540 |
+
public static function browserPost($url, $data, $callback, $param1 = null, $param2 = null, $param3 = null) {
|
| 5541 |
+
if (self::plugin('WebBrowser')) {
|
| 5542 |
+
$params = func_get_args();
|
| 5543 |
+
return self::callbackRun(array(self::$plugins, 'browserPost'), $params);
|
| 5544 |
+
} else {
|
| 5545 |
+
self::debug('WebBrowser plugin not available...');
|
| 5546 |
+
}
|
| 5547 |
+
}
|
| 5548 |
+
/**
|
| 5549 |
+
*
|
| 5550 |
+
* @param $ajaxSettings
|
| 5551 |
+
* @param $callback
|
| 5552 |
+
* @param $param1
|
| 5553 |
+
* @param $param2
|
| 5554 |
+
* @param $param3
|
| 5555 |
+
* @return phpQueryObject
|
| 5556 |
+
*/
|
| 5557 |
+
public static function browser($ajaxSettings, $callback, $param1 = null, $param2 = null, $param3 = null) {
|
| 5558 |
+
if (self::plugin('WebBrowser')) {
|
| 5559 |
+
$params = func_get_args();
|
| 5560 |
+
return self::callbackRun(array(self::$plugins, 'browser'), $params);
|
| 5561 |
+
} else {
|
| 5562 |
+
self::debug('WebBrowser plugin not available...');
|
| 5563 |
+
}
|
| 5564 |
+
}
|
| 5565 |
+
/**
|
| 5566 |
+
*
|
| 5567 |
+
* @param $code
|
| 5568 |
+
* @return string
|
| 5569 |
+
*/
|
| 5570 |
+
public static function php($code) {
|
| 5571 |
+
return self::code('php', $code);
|
| 5572 |
+
}
|
| 5573 |
+
/**
|
| 5574 |
+
*
|
| 5575 |
+
* @param $type
|
| 5576 |
+
* @param $code
|
| 5577 |
+
* @return string
|
| 5578 |
+
*/
|
| 5579 |
+
public static function code($type, $code) {
|
| 5580 |
+
return "<$type><!-- ".trim($code)." --></$type>";
|
| 5581 |
+
}
|
| 5582 |
+
|
| 5583 |
+
public static function __callStatic($method, $params) {
|
| 5584 |
+
return call_user_func_array(
|
| 5585 |
+
array(phpQuery::$plugins, $method),
|
| 5586 |
+
$params
|
| 5587 |
+
);
|
| 5588 |
+
}
|
| 5589 |
+
protected static function dataSetupNode($node, $documentID) {
|
| 5590 |
+
// search are return if alredy exists
|
| 5591 |
+
foreach(phpQuery::$documents[$documentID]->dataNodes as $dataNode) {
|
| 5592 |
+
if ($node->isSameNode($dataNode))
|
| 5593 |
+
return $dataNode;
|
| 5594 |
+
}
|
| 5595 |
+
// if doesn't, add it
|
| 5596 |
+
phpQuery::$documents[$documentID]->dataNodes[] = $node;
|
| 5597 |
+
return $node;
|
| 5598 |
+
}
|
| 5599 |
+
protected static function dataRemoveNode($node, $documentID) {
|
| 5600 |
+
// search are return if alredy exists
|
| 5601 |
+
foreach(phpQuery::$documents[$documentID]->dataNodes as $k => $dataNode) {
|
| 5602 |
+
if ($node->isSameNode($dataNode)) {
|
| 5603 |
+
unset(self::$documents[$documentID]->dataNodes[$k]);
|
| 5604 |
+
unset(self::$documents[$documentID]->data[ $dataNode->dataID ]);
|
| 5605 |
+
}
|
| 5606 |
+
}
|
| 5607 |
+
}
|
| 5608 |
+
public static function data($node, $name, $data, $documentID = null) {
|
| 5609 |
+
if (! $documentID)
|
| 5610 |
+
// TODO check if this works
|
| 5611 |
+
$documentID = self::getDocumentID($node);
|
| 5612 |
+
$document = phpQuery::$documents[$documentID];
|
| 5613 |
+
$node = self::dataSetupNode($node, $documentID);
|
| 5614 |
+
if (! isset($node->dataID))
|
| 5615 |
+
$node->dataID = ++phpQuery::$documents[$documentID]->uuid;
|
| 5616 |
+
$id = $node->dataID;
|
| 5617 |
+
if (! isset($document->data[$id]))
|
| 5618 |
+
$document->data[$id] = array();
|
| 5619 |
+
if (! is_null($data))
|
| 5620 |
+
$document->data[$id][$name] = $data;
|
| 5621 |
+
if ($name) {
|
| 5622 |
+
if (isset($document->data[$id][$name]))
|
| 5623 |
+
return $document->data[$id][$name];
|
| 5624 |
+
} else
|
| 5625 |
+
return $id;
|
| 5626 |
+
}
|
| 5627 |
+
public static function removeData($node, $name, $documentID) {
|
| 5628 |
+
if (! $documentID)
|
| 5629 |
+
// TODO check if this works
|
| 5630 |
+
$documentID = self::getDocumentID($node);
|
| 5631 |
+
$document = phpQuery::$documents[$documentID];
|
| 5632 |
+
$node = self::dataSetupNode($node, $documentID);
|
| 5633 |
+
$id = $node->dataID;
|
| 5634 |
+
if ($name) {
|
| 5635 |
+
if (isset($document->data[$id][$name]))
|
| 5636 |
+
unset($document->data[$id][$name]);
|
| 5637 |
+
$name = null;
|
| 5638 |
+
foreach($document->data[$id] as $name)
|
| 5639 |
+
break;
|
| 5640 |
+
if (! $name)
|
| 5641 |
+
self::removeData($node, $name, $documentID);
|
| 5642 |
+
} else {
|
| 5643 |
+
self::dataRemoveNode($node, $documentID);
|
| 5644 |
+
}
|
| 5645 |
+
}
|
| 5646 |
+
}
|
| 5647 |
+
/**
|
| 5648 |
+
* Plugins static namespace class.
|
| 5649 |
+
*
|
| 5650 |
+
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 5651 |
+
* @package phpQuery
|
| 5652 |
+
* @todo move plugin methods here (as statics)
|
| 5653 |
+
*/
|
| 5654 |
+
class phpQueryPlugins {
|
| 5655 |
+
public function __call($method, $args) {
|
| 5656 |
+
if (isset(phpQuery::$extendStaticMethods[$method])) {
|
| 5657 |
+
$return = call_user_func_array(
|
| 5658 |
+
phpQuery::$extendStaticMethods[$method],
|
| 5659 |
+
$args
|
| 5660 |
+
);
|
| 5661 |
+
} else if (isset(phpQuery::$pluginsStaticMethods[$method])) {
|
| 5662 |
+
$class = phpQuery::$pluginsStaticMethods[$method];
|
| 5663 |
+
$realClass = "phpQueryPlugin_$class";
|
| 5664 |
+
$return = call_user_func_array(
|
| 5665 |
+
array($realClass, $method),
|
| 5666 |
+
$args
|
| 5667 |
+
);
|
| 5668 |
+
return isset($return)
|
| 5669 |
+
? $return
|
| 5670 |
+
: $this;
|
| 5671 |
+
} else
|
| 5672 |
+
throw new Exception("Method '{$method}' doesnt exist");
|
| 5673 |
+
}
|
| 5674 |
+
}
|
| 5675 |
+
/**
|
| 5676 |
+
* Shortcut to phpQuery::pq($arg1, $context)
|
| 5677 |
+
* Chainable.
|
| 5678 |
+
*
|
| 5679 |
+
* @see phpQuery::pq()
|
| 5680 |
+
* @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
|
| 5681 |
+
* @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
|
| 5682 |
+
* @package phpQuery
|
| 5683 |
+
*/
|
| 5684 |
+
function pq($arg1, $context = null) {
|
| 5685 |
+
$args = func_get_args();
|
| 5686 |
+
return call_user_func_array(
|
| 5687 |
+
array('phpQuery', 'pq'),
|
| 5688 |
+
$args
|
| 5689 |
+
);
|
| 5690 |
+
}
|
| 5691 |
+
// add plugins dir and Zend framework to include path
|
| 5692 |
+
set_include_path(
|
| 5693 |
+
get_include_path()
|
| 5694 |
+
.PATH_SEPARATOR.dirname(__FILE__).'/phpQuery/'
|
| 5695 |
+
.PATH_SEPARATOR.dirname(__FILE__).'/phpQuery/plugins/'
|
| 5696 |
+
);
|
| 5697 |
+
// why ? no __call nor __get for statics in php...
|
| 5698 |
+
// XXX __callStatic will be available in PHP 5.3
|
| 5699 |
+
phpQuery::$plugins = new phpQueryPlugins();
|
| 5700 |
+
// include bootstrap file (personal library config)
|
| 5701 |
+
if (file_exists(dirname(__FILE__).'/phpQuery/bootstrap.php'))
|
| 5702 |
require_once dirname(__FILE__).'/phpQuery/bootstrap.php';
|
includes/wp-plugin-dev-classes/class-wp-meta-box-page.php
CHANGED
|
@@ -1,664 +1,664 @@
|
|
| 1 |
-
<?php defined( 'ABSPATH' ) OR die( 'No direct access.' );
|
| 2 |
-
if ( ! class_exists( 'WP_Meta_Box_Page_01' ) ):
|
| 3 |
-
/**
|
| 4 |
-
* WP_Meta_Box_Page_01 Class
|
| 5 |
-
*
|
| 6 |
-
* Creating a dashboard-like admin page with meta boxes
|
| 7 |
-
*
|
| 8 |
-
* Requires WordPress 3.0+ and PHP 5.2+
|
| 9 |
-
*
|
| 10 |
-
* Custom Actions:
|
| 11 |
-
* - load action params: (1) this object
|
| 12 |
-
* - meta_box_load
|
| 13 |
-
* - meta_box_load-[page_slug]
|
| 14 |
-
*
|
| 15 |
-
* Custom Filters:
|
| 16 |
-
* - page_title filter params: (1) default content, (2) this object
|
| 17 |
-
* - meta_box_page_title
|
| 18 |
-
* - meta_box_page_title-[page_slug]
|
| 19 |
-
*
|
| 20 |
-
* - page_content filter params: (1) default content, (2) this object
|
| 21 |
-
* - meta_box_page_content
|
| 22 |
-
* - meta_box_page_content-[page_slug]
|
| 23 |
-
*
|
| 24 |
-
* - screen_settings filter params: (1) default content, (2) current screen object, (3) this object
|
| 25 |
-
* - meta_box_screen_settings
|
| 26 |
-
* - meta_box_screen_settings-[page_slug]
|
| 27 |
-
*
|
| 28 |
-
* - contextual_help filter params: (1) default content, (2) current screen id, (3) current screen object, (4) this object
|
| 29 |
-
* - meta_box_contextual_help
|
| 30 |
-
* - meta_box_contextual_help-[page_slug]
|
| 31 |
-
*
|
| 32 |
-
* @version 0.1
|
| 33 |
-
* @author Victor Villaverde Laan
|
| 34 |
-
* @link http://www.freelancephp.net/
|
| 35 |
-
* @license Dual licensed under the MIT and GPL licenses
|
| 36 |
-
*/
|
| 37 |
-
class WP_Meta_Box_Page_01 {
|
| 38 |
-
|
| 39 |
-
/**
|
| 40 |
-
* Default settings
|
| 41 |
-
* @var array
|
| 42 |
-
*/
|
| 43 |
-
private static $default_settings = array(
|
| 44 |
-
// Page title
|
| 45 |
-
'page_title' => NULL, // Default will be set equal to $menu_title
|
| 46 |
-
|
| 47 |
-
// Menu title
|
| 48 |
-
'menu_title' => NULL, // Default will be set equal to $page_title
|
| 49 |
-
|
| 50 |
-
// Page slug
|
| 51 |
-
'page_slug' => NULL, // Default will be set to: sanitize_title_with_dashes( $menu_title )
|
| 52 |
-
|
| 53 |
-
// Default number of columns
|
| 54 |
-
'default_columns' => 2,
|
| 55 |
-
|
| 56 |
-
// Column widths
|
| 57 |
-
'column_widths' => array(
|
| 58 |
-
1 => array( 99 ),
|
| 59 |
-
2 => array( 49, 49 ),
|
| 60 |
-
3 => array( 32.33, 32.33, 32.33 ),
|
| 61 |
-
4 => array( 24, 24, 24, 24 ),
|
| 62 |
-
),
|
| 63 |
-
|
| 64 |
-
// Add page method
|
| 65 |
-
'add_page_method' => 'add_options_page', // OR: add_menu_page, add_object_page, add_submenu_page
|
| 66 |
-
|
| 67 |
-
// Extra params for the add_page_method
|
| 68 |
-
|
| 69 |
-
// Capability
|
| 70 |
-
'capability' => 'manage_options', // Optional for all methods
|
| 71 |
-
|
| 72 |
-
// Parent slug
|
| 73 |
-
'parent_slug' => NULL, // Nescessary when using "add_submenu_page"
|
| 74 |
-
|
| 75 |
-
// Url to the icon to be used for the menu ( around 20 x 20 pixels )
|
| 76 |
-
'icon_url' => NULL, // Only for "add_menu_page" or "add_object_page"
|
| 77 |
-
|
| 78 |
-
// The position in the menu order this menu should appear
|
| 79 |
-
'position' => NULL, // Only for "add_menu_page", default on bottom of the menu
|
| 80 |
-
);
|
| 81 |
-
|
| 82 |
-
/**
|
| 83 |
-
* Settings
|
| 84 |
-
* @var array
|
| 85 |
-
*/
|
| 86 |
-
private $settings = array();
|
| 87 |
-
|
| 88 |
-
/**
|
| 89 |
-
* Meta boxes
|
| 90 |
-
* @var array
|
| 91 |
-
*/
|
| 92 |
-
private $meta_boxes = array();
|
| 93 |
-
|
| 94 |
-
/**
|
| 95 |
-
* Pagehook ( will be set when page is created )
|
| 96 |
-
* @var string
|
| 97 |
-
*/
|
| 98 |
-
protected $pagehook = NULL;
|
| 99 |
-
|
| 100 |
-
|
| 101 |
-
/**
|
| 102 |
-
* Initialize
|
| 103 |
-
* @param array $settings Optional
|
| 104 |
-
*/
|
| 105 |
-
public function init( $settings = NULL, $load_callback = NULL ) {
|
| 106 |
-
// get settings of child class
|
| 107 |
-
$child_settings = $this->settings;
|
| 108 |
-
|
| 109 |
-
// set settings in 3 steps...
|
| 110 |
-
// (1) first set the default options
|
| 111 |
-
$this->set_setting( self::$default_settings );
|
| 112 |
-
|
| 113 |
-
// (2) set child settings
|
| 114 |
-
if ( ! empty( $child_settings ) )
|
| 115 |
-
$this->set_setting( $child_settings );
|
| 116 |
-
|
| 117 |
-
// (3) set param settings
|
| 118 |
-
if ( $settings !== NULL )
|
| 119 |
-
$this->set_setting( $settings );
|
| 120 |
-
|
| 121 |
-
// actions
|
| 122 |
-
add_action( 'admin_menu', array( $this, 'call_admin_menu' ) );
|
| 123 |
-
|
| 124 |
-
// add load action
|
| 125 |
-
if ( $load_callback !== NULL )
|
| 126 |
-
add_action( 'meta_box_load-'. $this->get_setting( 'page_slug' ), $load_callback );
|
| 127 |
-
}
|
| 128 |
-
|
| 129 |
-
/**
|
| 130 |
-
* Set default setting value
|
| 131 |
-
* @param mixed $key Also possible to give an array of key/value pairs
|
| 132 |
-
* @param mixed $value Optional
|
| 133 |
-
* @static
|
| 134 |
-
*/
|
| 135 |
-
public static function set_default_setting( $key, $value = NULL ) {
|
| 136 |
-
if ( is_array( $key ) ) {
|
| 137 |
-
foreach ( $key AS $k => $v )
|
| 138 |
-
self::set_default_setting( $k, $v );
|
| 139 |
-
} else {
|
| 140 |
-
self::$default_settings[ $key ] = $value;
|
| 141 |
-
}
|
| 142 |
-
}
|
| 143 |
-
|
| 144 |
-
/**
|
| 145 |
-
* Set setting value
|
| 146 |
-
* @param mixed $key Also possible to give an array of key/value pairs
|
| 147 |
-
* @param mixed $value Optional
|
| 148 |
-
* @return $this For chaining
|
| 149 |
-
*/
|
| 150 |
-
public function set_setting( $key, $value = NULL ) {
|
| 151 |
-
if ( is_array( $key ) ) {
|
| 152 |
-
foreach ( $key AS $k => $v )
|
| 153 |
-
$this->set_setting( $k, $v );
|
| 154 |
-
} else {
|
| 155 |
-
$this->settings[ $key ] = $value;
|
| 156 |
-
}
|
| 157 |
-
|
| 158 |
-
// auto-set related prop values
|
| 159 |
-
if ( $value !== NULL ) {
|
| 160 |
-
if ( $key == 'menu_title' AND $this->get_setting( 'page_title' ) === NULL )
|
| 161 |
-
$this->set_setting( 'page_title', $value );
|
| 162 |
-
|
| 163 |
-
if ( $key == 'page_title' AND $this->get_setting( 'menu_title' ) === NULL )
|
| 164 |
-
$this->set_setting( 'menu_title', $value );
|
| 165 |
-
|
| 166 |
-
if ( ( $key == 'menu_title' OR $key == 'page_title' ) AND $this->get_setting( 'page_slug' ) === NULL ) {
|
| 167 |
-
$new_val = $value;
|
| 168 |
-
$new_val = sanitize_title_with_dashes( $new_val );
|
| 169 |
-
$new_val = strtolower( $new_val );
|
| 170 |
-
|
| 171 |
-
// check for valid page_slug
|
| 172 |
-
if ( ! preg_match( '/^[a-z_-]+$/', $new_val ) ) {
|
| 173 |
-
$new_val = str_replace(
|
| 174 |
-
array( 1, 2, 3, 4, 5, 6, 7, 8, 9, 0 ),
|
| 175 |
-
array( 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j' ),
|
| 176 |
-
$new_val
|
| 177 |
-
);
|
| 178 |
-
|
| 179 |
-
$new_val = ereg_replace( '[^a-z_-]', '', $new_val );
|
| 180 |
-
}
|
| 181 |
-
|
| 182 |
-
$this->set_setting( 'page_slug', $new_val );
|
| 183 |
-
}
|
| 184 |
-
}
|
| 185 |
-
|
| 186 |
-
return $this;
|
| 187 |
-
}
|
| 188 |
-
|
| 189 |
-
/**
|
| 190 |
-
* Get setting value
|
| 191 |
-
* @param string $key Optional, when NULL will return array of all options
|
| 192 |
-
* @param mixed $default Optional, return default when key cannot be found OR is NULL
|
| 193 |
-
* @return mixed
|
| 194 |
-
*/
|
| 195 |
-
public function get_setting( $key = NULL, $default = NULL ) {
|
| 196 |
-
if ( $key === NULL )
|
| 197 |
-
return $this->settings;
|
| 198 |
-
|
| 199 |
-
if ( key_exists( $key, $this->settings ) )
|
| 200 |
-
return ( $this->settings[ $key ] === NULL ) ? $default : $this->settings[ $key ];
|
| 201 |
-
|
| 202 |
-
return $default;
|
| 203 |
-
}
|
| 204 |
-
|
| 205 |
-
/**
|
| 206 |
-
* Helper for adding a meta box
|
| 207 |
-
* @param string $title Title of the meta box
|
| 208 |
-
* @param string $callback Callback for the meta box content
|
| 209 |
-
* @param mixed $context Optional, add meta box to this column (normal = 1, side = 2, column3 = 3, column4 = 4)
|
| 210 |
-
* @param string $priority Optional, the priority within the context where the boxes should show ( 'high', 'core', 'default' or 'low' )
|
| 211 |
-
* @return $this For chaining
|
| 212 |
-
*/
|
| 213 |
-
public function add_meta_box( $title, $callback, $context = 'normal', $id = NULL, $priority = 'default' ) {
|
| 214 |
-
$this->meta_boxes[] = array(
|
| 215 |
-
'id' => $id,
|
| 216 |
-
'title' => $title,
|
| 217 |
-
'callback' => $callback,
|
| 218 |
-
'context' => $context,
|
| 219 |
-
'priority' => $priority,
|
| 220 |
-
);
|
| 221 |
-
|
| 222 |
-
return $this;
|
| 223 |
-
}
|
| 224 |
-
|
| 225 |
-
/**
|
| 226 |
-
* Add callback to "meta_box_load-[page]" action, only applied for this page/object
|
| 227 |
-
* @param mixed $callback Callback function
|
| 228 |
-
* @return $this
|
| 229 |
-
*/
|
| 230 |
-
public function add_load_action( $callback ) {
|
| 231 |
-
add_filter( 'meta_box_load-' . $this->get_setting( 'page_slug' ), $callback );
|
| 232 |
-
return $this;
|
| 233 |
-
}
|
| 234 |
-
|
| 235 |
-
/**
|
| 236 |
-
* Add callback to "meta_box_load" action, applied to all instances of this class
|
| 237 |
-
* @param mixed $callback Callback function
|
| 238 |
-
* @static
|
| 239 |
-
*/
|
| 240 |
-
public static function add_global_load_action( $callback ) {
|
| 241 |
-
add_filter( 'meta_box_load', $callback );
|
| 242 |
-
}
|
| 243 |
-
|
| 244 |
-
/**
|
| 245 |
-
* Add callback to "meta_box_screen_settings-[page]" filter, only applied for this page/object
|
| 246 |
-
* @param mixed $callback Callback function
|
| 247 |
-
* @return $this
|
| 248 |
-
*/
|
| 249 |
-
public function add_screen_settings_filter( $callback ) {
|
| 250 |
-
add_filter( 'meta_box_screen_settings-' . $this->get_setting( 'page_slug' ), $callback );
|
| 251 |
-
return $this;
|
| 252 |
-
}
|
| 253 |
-
|
| 254 |
-
/**
|
| 255 |
-
* Add callback to "meta_box_contextual_help-[page]" filter, only applied for this page/object
|
| 256 |
-
* @param mixed $callback Callback function
|
| 257 |
-
* @return $this
|
| 258 |
-
*/
|
| 259 |
-
public function add_contextual_help_filter( $callback ) {
|
| 260 |
-
add_filter( 'meta_box_contextual_help-' . $this->get_setting( 'page_slug' ), $callback );
|
| 261 |
-
return $this;
|
| 262 |
-
}
|
| 263 |
-
|
| 264 |
-
/**
|
| 265 |
-
* Add callback to "meta_box_page_title-[page]" filter, only applied for this page/object
|
| 266 |
-
* @param mixed $callback Callback function
|
| 267 |
-
* @return $this
|
| 268 |
-
*/
|
| 269 |
-
public function add_title_filter( $callback ) {
|
| 270 |
-
add_filter( 'meta_box_page_title-' . $this->get_setting( 'page_slug' ), $callback );
|
| 271 |
-
return $this;
|
| 272 |
-
}
|
| 273 |
-
|
| 274 |
-
/**
|
| 275 |
-
* Add callback to "meta_box_page_content" filter, applied to all instances of this class
|
| 276 |
-
* @param mixed $callback Callback function
|
| 277 |
-
* @static
|
| 278 |
-
*/
|
| 279 |
-
public static function add_global_title_filter( $callback ) {
|
| 280 |
-
add_filter( 'meta_box_page_title', $callback );
|
| 281 |
-
}
|
| 282 |
-
|
| 283 |
-
/**
|
| 284 |
-
* Add callback to "meta_box_page_content-[page]" filter, only applied for this page/object
|
| 285 |
-
* @param mixed $callback Callback function
|
| 286 |
-
* @return $this
|
| 287 |
-
*/
|
| 288 |
-
public function add_content_filter( $callback ) {
|
| 289 |
-
add_filter( 'meta_box_page_title-' . $this->get_setting( 'page_slug' ), $callback );
|
| 290 |
-
return $this;
|
| 291 |
-
}
|
| 292 |
-
|
| 293 |
-
/**
|
| 294 |
-
* Add callback to "meta_box_page_content" filter, applied to all instances of this class
|
| 295 |
-
* @param mixed $callback Callback function
|
| 296 |
-
* @static
|
| 297 |
-
*/
|
| 298 |
-
public static function add_global_content_filter( $callback ) {
|
| 299 |
-
add_filter( 'meta_box_page_title', $callback );
|
| 300 |
-
}
|
| 301 |
-
|
| 302 |
-
/**
|
| 303 |
-
* Admin menu callback
|
| 304 |
-
*/
|
| 305 |
-
public function call_admin_menu() {
|
| 306 |
-
// add page
|
| 307 |
-
switch ( $this->get_setting( 'add_page_method' ) ) {
|
| 308 |
-
case 'add_menu_page':
|
| 309 |
-
$this->pagehook = add_menu_page(
|
| 310 |
-
$this->get_setting( 'page_title' ),
|
| 311 |
-
$this->get_setting( 'menu_title' ),
|
| 312 |
-
$this->get_setting( 'capability' ),
|
| 313 |
-
$this->get_setting( 'page_slug' ),
|
| 314 |
-
array( $this, 'call_page_content' ),
|
| 315 |
-
$this->get_setting( 'icon_url' ),
|
| 316 |
-
$this->get_setting( 'position' )
|
| 317 |
-
);
|
| 318 |
-
break;
|
| 319 |
-
|
| 320 |
-
case 'add_object_page':
|
| 321 |
-
$this->pagehook = add_object_page(
|
| 322 |
-
$this->get_setting( 'page_title' ),
|
| 323 |
-
$this->get_setting( 'menu_title' ),
|
| 324 |
-
$this->get_setting( 'capability' ),
|
| 325 |
-
$this->get_setting( 'page_slug' ),
|
| 326 |
-
array( $this, 'call_page_content' ),
|
| 327 |
-
$this->get_setting( 'icon_url' )
|
| 328 |
-
);
|
| 329 |
-
break;
|
| 330 |
-
|
| 331 |
-
case 'add_submenu_page':
|
| 332 |
-
$this->pagehook = add_submenu_page(
|
| 333 |
-
$this->get_setting( 'parent_slug' ),
|
| 334 |
-
$this->get_setting( 'page_title' ),
|
| 335 |
-
$this->get_setting( 'menu_title' ),
|
| 336 |
-
$this->get_setting( 'capability' ),
|
| 337 |
-
$this->get_setting( 'page_slug' ),
|
| 338 |
-
array( $this, 'call_page_content' )
|
| 339 |
-
);
|
| 340 |
-
break;
|
| 341 |
-
|
| 342 |
-
case 'add_options_page':
|
| 343 |
-
default:
|
| 344 |
-
$this->pagehook = add_options_page(
|
| 345 |
-
$this->get_setting( 'page_title' ),
|
| 346 |
-
$this->get_setting( 'menu_title' ),
|
| 347 |
-
$this->get_setting( 'capability' ),
|
| 348 |
-
$this->get_setting( 'page_slug' ),
|
| 349 |
-
array( $this, 'call_page_content' )
|
| 350 |
-
);
|
| 351 |
-
break;
|
| 352 |
-
|
| 353 |
-
}
|
| 354 |
-
|
| 355 |
-
// execute action
|
| 356 |
-
do_action( 'meta_box_load', $this );
|
| 357 |
-
|
| 358 |
-
// load page
|
| 359 |
-
add_action( 'load-' . $this->pagehook, array( $this, 'call_load_page' ) );
|
| 360 |
-
}
|
| 361 |
-
|
| 362 |
-
/**
|
| 363 |
-
* Admin head callback
|
| 364 |
-
*/
|
| 365 |
-
public function call_admin_head() {
|
| 366 |
-
?>
|
| 367 |
-
<style type="text/css">
|
| 368 |
-
.postbox-container { padding-right:1%; float:left ; /* for WP < 3.1 */ }
|
| 369 |
-
.postbox-container .meta-box-sortables { min-height:200px; }
|
| 370 |
-
.postbox-container .postbox { min-width:0; }
|
| 371 |
-
.postbox-container .postbox .inside { margin:10px 0 ; padding:0 10px; } /* for WP < 3.2 */
|
| 372 |
-
</style>
|
| 373 |
-
<?php
|
| 374 |
-
}
|
| 375 |
-
|
| 376 |
-
/**
|
| 377 |
-
* Load page callback
|
| 378 |
-
*/
|
| 379 |
-
public function call_load_page() {
|
| 380 |
-
// execute action
|
| 381 |
-
do_action( 'meta_box_load-' . $this->get_setting( 'page_slug' ), $this );
|
| 382 |
-
|
| 383 |
-
// add script for meta boxes
|
| 384 |
-
wp_enqueue_script( 'postbox' );
|
| 385 |
-
|
| 386 |
-
// add to admin head
|
| 387 |
-
add_action( 'admin_head', array( $this, 'call_admin_head' ) );
|
| 388 |
-
|
| 389 |
-
// add screen settings filter
|
| 390 |
-
add_filter( 'screen_settings', array( $this, 'call_screen_settings') );
|
| 391 |
-
|
| 392 |
-
// add help text
|
| 393 |
-
add_filter( 'contextual_help', array( $this, 'call_contextual_help' ) );
|
| 394 |
-
|
| 395 |
-
// columns
|
| 396 |
-
if ( function_exists( 'add_screen_option' ) ) {
|
| 397 |
-
$count = count( $this->get_setting( 'column_widths' ) );
|
| 398 |
-
|
| 399 |
-
add_screen_option( 'layout_columns', array(
|
| 400 |
-
'max' => $count,
|
| 401 |
-
'default' => min( $count, $this->get_setting( 'default_columns' ) )
|
| 402 |
-
));
|
| 403 |
-
}
|
| 404 |
-
|
| 405 |
-
// add meta boxes
|
| 406 |
-
$nr = 0;
|
| 407 |
-
foreach ( $this->meta_boxes AS $box ) {
|
| 408 |
-
$title = $box[ 'title' ];
|
| 409 |
-
$id = ( isset( $box[ 'id' ] ) ) ? $box[ 'id' ] : sanitize_title_with_dashes( $title .'-'. ++$nr, 'meta-box-' . $nr );
|
| 410 |
-
$callback = ( is_string( $box[ 'callback' ] ) && method_exists( $this, $box[ 'callback' ] ) ) ? array( $this, $box[ 'callback' ] ) : $box[ 'callback' ];
|
| 411 |
-
$context = $box[ 'context' ];
|
| 412 |
-
$priority = $box[ 'priority' ];
|
| 413 |
-
|
| 414 |
-
// set context
|
| 415 |
-
if ( $context == 2 OR strtolower( $context ) == 'side' ) {
|
| 416 |
-
$context = 'side';
|
| 417 |
-
} elseif ( $context == 3 OR strtolower( $context ) == 'column3' ) {
|
| 418 |
-
$context = 'column3';
|
| 419 |
-
} elseif ( $context == 4 OR strtolower( $context ) == 'column4' ) {
|
| 420 |
-
$context = 'column4';
|
| 421 |
-
} else { // default
|
| 422 |
-
$context = 'normal';
|
| 423 |
-
}
|
| 424 |
-
|
| 425 |
-
// add meta box
|
| 426 |
-
add_meta_box( $id, $title, $callback, $this->pagehook, $context, $priority ); // $callback_args, doesn't seem to work
|
| 427 |
-
}
|
| 428 |
-
}
|
| 429 |
-
|
| 430 |
-
/**
|
| 431 |
-
* Screen settings (callback)
|
| 432 |
-
* @param string $content
|
| 433 |
-
* @return string
|
| 434 |
-
*/
|
| 435 |
-
public function call_screen_settings( $content ) {
|
| 436 |
-
if ( self::get_current_screen()->id == convert_to_screen( $this->pagehook )->id ) {
|
| 437 |
-
// apply filters for this meta box page
|
| 438 |
-
$content = apply_filters( 'meta_box_screen_settings-' . $this->get_setting( 'page_slug' ), $content, $this->get_current_screen(), $this );
|
| 439 |
-
}
|
| 440 |
-
|
| 441 |
-
return $content;
|
| 442 |
-
}
|
| 443 |
-
|
| 444 |
-
/**
|
| 445 |
-
* Contextual help (callback)
|
| 446 |
-
* @param string $content
|
| 447 |
-
* @return string
|
| 448 |
-
*/
|
| 449 |
-
public function call_contextual_help( $content ) {
|
| 450 |
-
$current_screen = $this->get_current_screen();
|
| 451 |
-
|
| 452 |
-
if ( $current_screen->id == convert_to_screen( $this->pagehook )->id ) {
|
| 453 |
-
// apply filters for this meta box page
|
| 454 |
-
$content = apply_filters( 'meta_box_contextual_help-' . $this->get_setting( 'page_slug' ), $content, $current_screen->id, $current_screen, $this );
|
| 455 |
-
}
|
| 456 |
-
|
| 457 |
-
return $content;
|
| 458 |
-
}
|
| 459 |
-
|
| 460 |
-
/**
|
| 461 |
-
* Display admin page content (callback)
|
| 462 |
-
*/
|
| 463 |
-
public function call_page_content() {
|
| 464 |
-
echo '<div class="wrap">';
|
| 465 |
-
|
| 466 |
-
// page title
|
| 467 |
-
$meta_boxes_page_title = apply_filters( 'meta_box_page_title', $this->get_page_title(), $this );
|
| 468 |
-
echo apply_filters( 'meta_box_page_title-' . $this->get_setting( 'page_slug' ), $meta_boxes_page_title, $this );
|
| 469 |
-
|
| 470 |
-
// page content
|
| 471 |
-
$meta_boxes_content = apply_filters( 'meta_box_page_content', $this->get_page_content(), $this );
|
| 472 |
-
echo apply_filters( 'meta_box_page_content-' . $this->get_setting( 'page_slug' ), $meta_boxes_content, $this );
|
| 473 |
-
|
| 474 |
-
echo '</div>';
|
| 475 |
-
}
|
| 476 |
-
|
| 477 |
-
/**
|
| 478 |
-
* Get page title
|
| 479 |
-
* Can be changed by using adding filters, see add_title_filter()
|
| 480 |
-
* @return string
|
| 481 |
-
*/
|
| 482 |
-
private function get_page_title() {
|
| 483 |
-
return '<h2>' . get_admin_page_title() . '</h2>';
|
| 484 |
-
}
|
| 485 |
-
|
| 486 |
-
/**
|
| 487 |
-
* Get meta boxes content. Can be changed by adding filters, see add_content_filter()
|
| 488 |
-
* @return string
|
| 489 |
-
*/
|
| 490 |
-
private function get_page_content() {
|
| 491 |
-
return self::get_ob_callback( array( $this, 'show_meta_boxes' ) );
|
| 492 |
-
}
|
| 493 |
-
|
| 494 |
-
/**
|
| 495 |
-
* Display meta boxes content
|
| 496 |
-
*/
|
| 497 |
-
private function show_meta_boxes() {
|
| 498 |
-
$opt_column_widths = $this->get_setting( 'column_widths' );
|
| 499 |
-
$hide2 = $hide3 = $hide4 = '';
|
| 500 |
-
switch ( self::get_screen_layout_columns( $this->get_setting( 'default_columns' ) ) ) {
|
| 501 |
-
case 4:
|
| 502 |
-
$column_widths = $opt_column_widths[ 4 ];
|
| 503 |
-
break;
|
| 504 |
-
case 3:
|
| 505 |
-
$column_widths = $opt_column_widths[ 3 ];
|
| 506 |
-
$hide4 = 'display:none;';
|
| 507 |
-
break;
|
| 508 |
-
case 2:
|
| 509 |
-
$column_widths = $opt_column_widths[ 2 ];
|
| 510 |
-
$hide3 = $hide4 = 'display:none;';
|
| 511 |
-
break;
|
| 512 |
-
default:
|
| 513 |
-
$column_widths = $opt_column_widths[ 1 ];
|
| 514 |
-
$hide2 = $hide3 = $hide4 = 'display:none;';
|
| 515 |
-
}
|
| 516 |
-
|
| 517 |
-
$column_widths = array_pad( $column_widths, 4, 0 );
|
| 518 |
-
?>
|
| 519 |
-
<div id='<?php echo $this->pagehook ?>-widgets' class='metabox-holder'>
|
| 520 |
-
<div class='postbox-container' style='width:<?php echo $column_widths[0] ?>%'>
|
| 521 |
-
<?php do_meta_boxes( $this->pagehook, 'normal', '' ); ?>
|
| 522 |
-
</div>
|
| 523 |
-
|
| 524 |
-
<div class='postbox-container' style='<?php echo $hide2 ?>width:<?php echo $column_widths[1] ?>%'>
|
| 525 |
-
<?php do_meta_boxes( $this->pagehook, 'side', '' ); ?>
|
| 526 |
-
</div>
|
| 527 |
-
|
| 528 |
-
<div class='postbox-container' style='<?php echo $hide3 ?>width:<?php echo $column_widths[2] ?>%'>
|
| 529 |
-
<?php do_meta_boxes( $this->pagehook, 'column3', '' ); ?>
|
| 530 |
-
</div>
|
| 531 |
-
|
| 532 |
-
<div class='postbox-container' style='<?php echo $hide4 ?>width:<?php echo $column_widths[3] ?>%'>
|
| 533 |
-
<?php do_meta_boxes( $this->pagehook, 'column4', '' ); ?>
|
| 534 |
-
</div>
|
| 535 |
-
</div>
|
| 536 |
-
|
| 537 |
-
<form style="display:none" method="get" action="">
|
| 538 |
-
<?php wp_nonce_field( 'closedpostboxes', 'closedpostboxesnonce', false ); ?>
|
| 539 |
-
<?php wp_nonce_field( 'meta-box-order', 'meta-box-order-nonce', false ); ?>
|
| 540 |
-
</form>
|
| 541 |
-
|
| 542 |
-
<script type="text/javascript">
|
| 543 |
-
//<![CDATA[
|
| 544 |
-
jQuery( document ).ready( function( $ ){
|
| 545 |
-
var columnWidths = <?php echo json_encode( $opt_column_widths ) ?>,
|
| 546 |
-
$boxes = $( '.postbox-container' ),
|
| 547 |
-
setColumnWidths = function () {
|
| 548 |
-
var c = $( 'input[name="screen_columns"]:checked' ).val();
|
| 549 |
-
|
| 550 |
-
// first hide all boxes
|
| 551 |
-
$boxes.hide();
|
| 552 |
-
|
| 553 |
-
// set width and show boxes
|
| 554 |
-
for ( var x = 0; x < columnWidths[ c ].length; x++ ) {
|
| 555 |
-
$boxes.eq( x )
|
| 556 |
-
.css( 'width', columnWidths[ c ][ x ]+ '%' )
|
| 557 |
-
.show();
|
| 558 |
-
}
|
| 559 |
-
};
|
| 560 |
-
|
| 561 |
-
// radio screen columns
|
| 562 |
-
$( 'input[name="screen_columns"]' )
|
| 563 |
-
.click(function(){
|
| 564 |
-
if ( $( 'input[name="screen_columns"]:checked' ).val() == $( this ).val() ) {
|
| 565 |
-
setTimeout(function(){
|
| 566 |
-
setColumnWidths();
|
| 567 |
-
}, 1 );
|
| 568 |
-
}
|
| 569 |
-
})
|
| 570 |
-
.change(function( e ){
|
| 571 |
-
setColumnWidths();
|
| 572 |
-
|
| 573 |
-
// prevent
|
| 574 |
-
e.stopImmediatePropagation();
|
| 575 |
-
});
|
| 576 |
-
|
| 577 |
-
// trigger change event of selected column
|
| 578 |
-
$( 'input[name="screen_columns"]:checked' ).change();
|
| 579 |
-
|
| 580 |
-
<?php if ( self::wp_version( '3.2', '<' ) ): ?>
|
| 581 |
-
// for WP < 3.2
|
| 582 |
-
|
| 583 |
-
// close postboxes that should be closed
|
| 584 |
-
$( '.if-js-closed' ).removeClass( 'if-js-closed' ).addClass( 'closed' );
|
| 585 |
-
|
| 586 |
-
// Loading saved screen settings
|
| 587 |
-
postboxes.add_postbox_toggles( '<?php echo $this->pagehook ?>' );
|
| 588 |
-
<?php endif; ?>
|
| 589 |
-
});
|
| 590 |
-
//]]>
|
| 591 |
-
</script>
|
| 592 |
-
<?php
|
| 593 |
-
}
|
| 594 |
-
|
| 595 |
-
/**
|
| 596 |
-
* Static helpers
|
| 597 |
-
*/
|
| 598 |
-
|
| 599 |
-
/**
|
| 600 |
-
* Get content displayed by given callback
|
| 601 |
-
* @param mixed $callback
|
| 602 |
-
* @return string
|
| 603 |
-
* @static
|
| 604 |
-
*/
|
| 605 |
-
public static function get_ob_callback( $callback ) {
|
| 606 |
-
// start output buffer
|
| 607 |
-
ob_start();
|
| 608 |
-
|
| 609 |
-
// call callback
|
| 610 |
-
call_user_func( $callback );
|
| 611 |
-
|
| 612 |
-
// get the view content
|
| 613 |
-
$content = ob_get_contents();
|
| 614 |
-
|
| 615 |
-
// clean output buffer
|
| 616 |
-
ob_end_clean();
|
| 617 |
-
|
| 618 |
-
return $content;
|
| 619 |
-
}
|
| 620 |
-
|
| 621 |
-
/**
|
| 622 |
-
* Get current version of WP or compare versions
|
| 623 |
-
* @global string $wp_version
|
| 624 |
-
* @return mixed
|
| 625 |
-
* @static
|
| 626 |
-
*/
|
| 627 |
-
public static function wp_version( $compare_version = NULL, $operator = NULL ) {
|
| 628 |
-
global $wp_version;
|
| 629 |
-
$cur_wp_version = preg_replace( '/-.*$/', '', $wp_version );
|
| 630 |
-
|
| 631 |
-
if ( $compare_version === NULL )
|
| 632 |
-
return $cur_wp_version;
|
| 633 |
-
|
| 634 |
-
// check comparison
|
| 635 |
-
return version_compare( $cur_wp_version, $compare_version, $operator );
|
| 636 |
-
}
|
| 637 |
-
|
| 638 |
-
/**
|
| 639 |
-
* Return global WP value of $screen_layout_columns
|
| 640 |
-
* @global integer $screen_layout_columns
|
| 641 |
-
* @return integer
|
| 642 |
-
* @static
|
| 643 |
-
*/
|
| 644 |
-
public static function get_screen_layout_columns( $default = NULL ) {
|
| 645 |
-
global $screen_layout_columns;
|
| 646 |
-
return ( empty( $screen_layout_columns ) ) ? $default : $screen_layout_columns;
|
| 647 |
-
}
|
| 648 |
-
|
| 649 |
-
/**
|
| 650 |
-
* Return global WP value of $current_screen
|
| 651 |
-
* @global string $current_screen
|
| 652 |
-
* @return string
|
| 653 |
-
* @static
|
| 654 |
-
*/
|
| 655 |
-
public static function get_current_screen( $default = NULL ) {
|
| 656 |
-
global $current_screen;
|
| 657 |
-
return $current_screen;
|
| 658 |
-
}
|
| 659 |
-
|
| 660 |
-
} // End WP_Meta_Box_Page_01 Class
|
| 661 |
-
|
| 662 |
-
endif;
|
| 663 |
-
|
| 664 |
/* ommit PHP closing tag, to prevent unwanted whitespace at the end of the parts generated by the included files */
|
| 1 |
+
<?php defined( 'ABSPATH' ) OR die( 'No direct access.' );
|
| 2 |
+
if ( ! class_exists( 'WP_Meta_Box_Page_01' ) ):
|
| 3 |
+
/**
|
| 4 |
+
* WP_Meta_Box_Page_01 Class
|
| 5 |
+
*
|
| 6 |
+
* Creating a dashboard-like admin page with meta boxes
|
| 7 |
+
*
|
| 8 |
+
* Requires WordPress 3.0+ and PHP 5.2+
|
| 9 |
+
*
|
| 10 |
+
* Custom Actions:
|
| 11 |
+
* - load action params: (1) this object
|
| 12 |
+
* - meta_box_load
|
| 13 |
+
* - meta_box_load-[page_slug]
|
| 14 |
+
*
|
| 15 |
+
* Custom Filters:
|
| 16 |
+
* - page_title filter params: (1) default content, (2) this object
|
| 17 |
+
* - meta_box_page_title
|
| 18 |
+
* - meta_box_page_title-[page_slug]
|
| 19 |
+
*
|
| 20 |
+
* - page_content filter params: (1) default content, (2) this object
|
| 21 |
+
* - meta_box_page_content
|
| 22 |
+
* - meta_box_page_content-[page_slug]
|
| 23 |
+
*
|
| 24 |
+
* - screen_settings filter params: (1) default content, (2) current screen object, (3) this object
|
| 25 |
+
* - meta_box_screen_settings
|
| 26 |
+
* - meta_box_screen_settings-[page_slug]
|
| 27 |
+
*
|
| 28 |
+
* - contextual_help filter params: (1) default content, (2) current screen id, (3) current screen object, (4) this object
|
| 29 |
+
* - meta_box_contextual_help
|
| 30 |
+
* - meta_box_contextual_help-[page_slug]
|
| 31 |
+
*
|
| 32 |
+
* @version 0.1
|
| 33 |
+
* @author Victor Villaverde Laan
|
| 34 |
+
* @link http://www.freelancephp.net/
|
| 35 |
+
* @license Dual licensed under the MIT and GPL licenses
|
| 36 |
+
*/
|
| 37 |
+
class WP_Meta_Box_Page_01 {
|
| 38 |
+
|
| 39 |
+
/**
|
| 40 |
+
* Default settings
|
| 41 |
+
* @var array
|
| 42 |
+
*/
|
| 43 |
+
private static $default_settings = array(
|
| 44 |
+
// Page title
|
| 45 |
+
'page_title' => NULL, // Default will be set equal to $menu_title
|
| 46 |
+
|
| 47 |
+
// Menu title
|
| 48 |
+
'menu_title' => NULL, // Default will be set equal to $page_title
|
| 49 |
+
|
| 50 |
+
// Page slug
|
| 51 |
+
'page_slug' => NULL, // Default will be set to: sanitize_title_with_dashes( $menu_title )
|
| 52 |
+
|
| 53 |
+
// Default number of columns
|
| 54 |
+
'default_columns' => 2,
|
| 55 |
+
|
| 56 |
+
// Column widths
|
| 57 |
+
'column_widths' => array(
|
| 58 |
+
1 => array( 99 ),
|
| 59 |
+
2 => array( 49, 49 ),
|
| 60 |
+
3 => array( 32.33, 32.33, 32.33 ),
|
| 61 |
+
4 => array( 24, 24, 24, 24 ),
|
| 62 |
+
),
|
| 63 |
+
|
| 64 |
+
// Add page method
|
| 65 |
+
'add_page_method' => 'add_options_page', // OR: add_menu_page, add_object_page, add_submenu_page
|
| 66 |
+
|
| 67 |
+
// Extra params for the add_page_method
|
| 68 |
+
|
| 69 |
+
// Capability
|
| 70 |
+
'capability' => 'manage_options', // Optional for all methods
|
| 71 |
+
|
| 72 |
+
// Parent slug
|
| 73 |
+
'parent_slug' => NULL, // Nescessary when using "add_submenu_page"
|
| 74 |
+
|
| 75 |
+
// Url to the icon to be used for the menu ( around 20 x 20 pixels )
|
| 76 |
+
'icon_url' => NULL, // Only for "add_menu_page" or "add_object_page"
|
| 77 |
+
|
| 78 |
+
// The position in the menu order this menu should appear
|
| 79 |
+
'position' => NULL, // Only for "add_menu_page", default on bottom of the menu
|
| 80 |
+
);
|
| 81 |
+
|
| 82 |
+
/**
|
| 83 |
+
* Settings
|
| 84 |
+
* @var array
|
| 85 |
+
*/
|
| 86 |
+
private $settings = array();
|
| 87 |
+
|
| 88 |
+
/**
|
| 89 |
+
* Meta boxes
|
| 90 |
+
* @var array
|
| 91 |
+
*/
|
| 92 |
+
private $meta_boxes = array();
|
| 93 |
+
|
| 94 |
+
/**
|
| 95 |
+
* Pagehook ( will be set when page is created )
|
| 96 |
+
* @var string
|
| 97 |
+
*/
|
| 98 |
+
protected $pagehook = NULL;
|
| 99 |
+
|
| 100 |
+
|
| 101 |
+
/**
|
| 102 |
+
* Initialize
|
| 103 |
+
* @param array $settings Optional
|
| 104 |
+
*/
|
| 105 |
+
public function init( $settings = NULL, $load_callback = NULL ) {
|
| 106 |
+
// get settings of child class
|
| 107 |
+
$child_settings = $this->settings;
|
| 108 |
+
|
| 109 |
+
// set settings in 3 steps...
|
| 110 |
+
// (1) first set the default options
|
| 111 |
+
$this->set_setting( self::$default_settings );
|
| 112 |
+
|
| 113 |
+
// (2) set child settings
|
| 114 |
+
if ( ! empty( $child_settings ) )
|
| 115 |
+
$this->set_setting( $child_settings );
|
| 116 |
+
|
| 117 |
+
// (3) set param settings
|
| 118 |
+
if ( $settings !== NULL )
|
| 119 |
+
$this->set_setting( $settings );
|
| 120 |
+
|
| 121 |
+
// actions
|
| 122 |
+
add_action( 'admin_menu', array( $this, 'call_admin_menu' ) );
|
| 123 |
+
|
| 124 |
+
// add load action
|
| 125 |
+
if ( $load_callback !== NULL )
|
| 126 |
+
add_action( 'meta_box_load-'. $this->get_setting( 'page_slug' ), $load_callback );
|
| 127 |
+
}
|
| 128 |
+
|
| 129 |
+
/**
|
| 130 |
+
* Set default setting value
|
| 131 |
+
* @param mixed $key Also possible to give an array of key/value pairs
|
| 132 |
+
* @param mixed $value Optional
|
| 133 |
+
* @static
|
| 134 |
+
*/
|
| 135 |
+
public static function set_default_setting( $key, $value = NULL ) {
|
| 136 |
+
if ( is_array( $key ) ) {
|
| 137 |
+
foreach ( $key AS $k => $v )
|
| 138 |
+
self::set_default_setting( $k, $v );
|
| 139 |
+
} else {
|
| 140 |
+
self::$default_settings[ $key ] = $value;
|
| 141 |
+
}
|
| 142 |
+
}
|
| 143 |
+
|
| 144 |
+
/**
|
| 145 |
+
* Set setting value
|
| 146 |
+
* @param mixed $key Also possible to give an array of key/value pairs
|
| 147 |
+
* @param mixed $value Optional
|
| 148 |
+
* @return $this For chaining
|
| 149 |
+
*/
|
| 150 |
+
public function set_setting( $key, $value = NULL ) {
|
| 151 |
+
if ( is_array( $key ) ) {
|
| 152 |
+
foreach ( $key AS $k => $v )
|
| 153 |
+
$this->set_setting( $k, $v );
|
| 154 |
+
} else {
|
| 155 |
+
$this->settings[ $key ] = $value;
|
| 156 |
+
}
|
| 157 |
+
|
| 158 |
+
// auto-set related prop values
|
| 159 |
+
if ( $value !== NULL ) {
|
| 160 |
+
if ( $key == 'menu_title' AND $this->get_setting( 'page_title' ) === NULL )
|
| 161 |
+
$this->set_setting( 'page_title', $value );
|
| 162 |
+
|
| 163 |
+
if ( $key == 'page_title' AND $this->get_setting( 'menu_title' ) === NULL )
|
| 164 |
+
$this->set_setting( 'menu_title', $value );
|
| 165 |
+
|
| 166 |
+
if ( ( $key == 'menu_title' OR $key == 'page_title' ) AND $this->get_setting( 'page_slug' ) === NULL ) {
|
| 167 |
+
$new_val = $value;
|
| 168 |
+
$new_val = sanitize_title_with_dashes( $new_val );
|
| 169 |
+
$new_val = strtolower( $new_val );
|
| 170 |
+
|
| 171 |
+
// check for valid page_slug
|
| 172 |
+
if ( ! preg_match( '/^[a-z_-]+$/', $new_val ) ) {
|
| 173 |
+
$new_val = str_replace(
|
| 174 |
+
array( 1, 2, 3, 4, 5, 6, 7, 8, 9, 0 ),
|
| 175 |
+
array( 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j' ),
|
| 176 |
+
$new_val
|
| 177 |
+
);
|
| 178 |
+
|
| 179 |
+
$new_val = ereg_replace( '[^a-z_-]', '', $new_val );
|
| 180 |
+
}
|
| 181 |
+
|
| 182 |
+
$this->set_setting( 'page_slug', $new_val );
|
| 183 |
+
}
|
| 184 |
+
}
|
| 185 |
+
|
| 186 |
+
return $this;
|
| 187 |
+
}
|
| 188 |
+
|
| 189 |
+
/**
|
| 190 |
+
* Get setting value
|
| 191 |
+
* @param string $key Optional, when NULL will return array of all options
|
| 192 |
+
* @param mixed $default Optional, return default when key cannot be found OR is NULL
|
| 193 |
+
* @return mixed
|
| 194 |
+
*/
|
| 195 |
+
public function get_setting( $key = NULL, $default = NULL ) {
|
| 196 |
+
if ( $key === NULL )
|
| 197 |
+
return $this->settings;
|
| 198 |
+
|
| 199 |
+
if ( key_exists( $key, $this->settings ) )
|
| 200 |
+
return ( $this->settings[ $key ] === NULL ) ? $default : $this->settings[ $key ];
|
| 201 |
+
|
| 202 |
+
return $default;
|
| 203 |
+
}
|
| 204 |
+
|
| 205 |
+
/**
|
| 206 |
+
* Helper for adding a meta box
|
| 207 |
+
* @param string $title Title of the meta box
|
| 208 |
+
* @param string $callback Callback for the meta box content
|
| 209 |
+
* @param mixed $context Optional, add meta box to this column (normal = 1, side = 2, column3 = 3, column4 = 4)
|
| 210 |
+
* @param string $priority Optional, the priority within the context where the boxes should show ( 'high', 'core', 'default' or 'low' )
|
| 211 |
+
* @return $this For chaining
|
| 212 |
+
*/
|
| 213 |
+
public function add_meta_box( $title, $callback, $context = 'normal', $id = NULL, $priority = 'default' ) {
|
| 214 |
+
$this->meta_boxes[] = array(
|
| 215 |
+
'id' => $id,
|
| 216 |
+
'title' => $title,
|
| 217 |
+
'callback' => $callback,
|
| 218 |
+
'context' => $context,
|
| 219 |
+
'priority' => $priority,
|
| 220 |
+
);
|
| 221 |
+
|
| 222 |
+
return $this;
|
| 223 |
+
}
|
| 224 |
+
|
| 225 |
+
/**
|
| 226 |
+
* Add callback to "meta_box_load-[page]" action, only applied for this page/object
|
| 227 |
+
* @param mixed $callback Callback function
|
| 228 |
+
* @return $this
|
| 229 |
+
*/
|
| 230 |
+
public function add_load_action( $callback ) {
|
| 231 |
+
add_filter( 'meta_box_load-' . $this->get_setting( 'page_slug' ), $callback );
|
| 232 |
+
return $this;
|
| 233 |
+
}
|
| 234 |
+
|
| 235 |
+
/**
|
| 236 |
+
* Add callback to "meta_box_load" action, applied to all instances of this class
|
| 237 |
+
* @param mixed $callback Callback function
|
| 238 |
+
* @static
|
| 239 |
+
*/
|
| 240 |
+
public static function add_global_load_action( $callback ) {
|
| 241 |
+
add_filter( 'meta_box_load', $callback );
|
| 242 |
+
}
|
| 243 |
+
|
| 244 |
+
/**
|
| 245 |
+
* Add callback to "meta_box_screen_settings-[page]" filter, only applied for this page/object
|
| 246 |
+
* @param mixed $callback Callback function
|
| 247 |
+
* @return $this
|
| 248 |
+
*/
|
| 249 |
+
public function add_screen_settings_filter( $callback ) {
|
| 250 |
+
add_filter( 'meta_box_screen_settings-' . $this->get_setting( 'page_slug' ), $callback );
|
| 251 |
+
return $this;
|
| 252 |
+
}
|
| 253 |
+
|
| 254 |
+
/**
|
| 255 |
+
* Add callback to "meta_box_contextual_help-[page]" filter, only applied for this page/object
|
| 256 |
+
* @param mixed $callback Callback function
|
| 257 |
+
* @return $this
|
| 258 |
+
*/
|
| 259 |
+
public function add_contextual_help_filter( $callback ) {
|
| 260 |
+
add_filter( 'meta_box_contextual_help-' . $this->get_setting( 'page_slug' ), $callback );
|
| 261 |
+
return $this;
|
| 262 |
+
}
|
| 263 |
+
|
| 264 |
+
/**
|
| 265 |
+
* Add callback to "meta_box_page_title-[page]" filter, only applied for this page/object
|
| 266 |
+
* @param mixed $callback Callback function
|
| 267 |
+
* @return $this
|
| 268 |
+
*/
|
| 269 |
+
public function add_title_filter( $callback ) {
|
| 270 |
+
add_filter( 'meta_box_page_title-' . $this->get_setting( 'page_slug' ), $callback );
|
| 271 |
+
return $this;
|
| 272 |
+
}
|
| 273 |
+
|
| 274 |
+
/**
|
| 275 |
+
* Add callback to "meta_box_page_content" filter, applied to all instances of this class
|
| 276 |
+
* @param mixed $callback Callback function
|
| 277 |
+
* @static
|
| 278 |
+
*/
|
| 279 |
+
public static function add_global_title_filter( $callback ) {
|
| 280 |
+
add_filter( 'meta_box_page_title', $callback );
|
| 281 |
+
}
|
| 282 |
+
|
| 283 |
+
/**
|
| 284 |
+
* Add callback to "meta_box_page_content-[page]" filter, only applied for this page/object
|
| 285 |
+
* @param mixed $callback Callback function
|
| 286 |
+
* @return $this
|
| 287 |
+
*/
|
| 288 |
+
public function add_content_filter( $callback ) {
|
| 289 |
+
add_filter( 'meta_box_page_title-' . $this->get_setting( 'page_slug' ), $callback );
|
| 290 |
+
return $this;
|
| 291 |
+
}
|
| 292 |
+
|
| 293 |
+
/**
|
| 294 |
+
* Add callback to "meta_box_page_content" filter, applied to all instances of this class
|
| 295 |
+
* @param mixed $callback Callback function
|
| 296 |
+
* @static
|
| 297 |
+
*/
|
| 298 |
+
public static function add_global_content_filter( $callback ) {
|
| 299 |
+
add_filter( 'meta_box_page_title', $callback );
|
| 300 |
+
}
|
| 301 |
+
|
| 302 |
+
/**
|
| 303 |
+
* Admin menu callback
|
| 304 |
+
*/
|
| 305 |
+
public function call_admin_menu() {
|
| 306 |
+
// add page
|
| 307 |
+
switch ( $this->get_setting( 'add_page_method' ) ) {
|
| 308 |
+
case 'add_menu_page':
|
| 309 |
+
$this->pagehook = add_menu_page(
|
| 310 |
+
$this->get_setting( 'page_title' ),
|
| 311 |
+
$this->get_setting( 'menu_title' ),
|
| 312 |
+
$this->get_setting( 'capability' ),
|
| 313 |
+
$this->get_setting( 'page_slug' ),
|
| 314 |
+
array( $this, 'call_page_content' ),
|
| 315 |
+
$this->get_setting( 'icon_url' ),
|
| 316 |
+
$this->get_setting( 'position' )
|
| 317 |
+
);
|
| 318 |
+
break;
|
| 319 |
+
|
| 320 |
+
case 'add_object_page':
|
| 321 |
+
$this->pagehook = add_object_page(
|
| 322 |
+
$this->get_setting( 'page_title' ),
|
| 323 |
+
$this->get_setting( 'menu_title' ),
|
| 324 |
+
$this->get_setting( 'capability' ),
|
| 325 |
+
$this->get_setting( 'page_slug' ),
|
| 326 |
+
array( $this, 'call_page_content' ),
|
| 327 |
+
$this->get_setting( 'icon_url' )
|
| 328 |
+
);
|
| 329 |
+
break;
|
| 330 |
+
|
| 331 |
+
case 'add_submenu_page':
|
| 332 |
+
$this->pagehook = add_submenu_page(
|
| 333 |
+
$this->get_setting( 'parent_slug' ),
|
| 334 |
+
$this->get_setting( 'page_title' ),
|
| 335 |
+
$this->get_setting( 'menu_title' ),
|
| 336 |
+
$this->get_setting( 'capability' ),
|
| 337 |
+
$this->get_setting( 'page_slug' ),
|
| 338 |
+
array( $this, 'call_page_content' )
|
| 339 |
+
);
|
| 340 |
+
break;
|
| 341 |
+
|
| 342 |
+
case 'add_options_page':
|
| 343 |
+
default:
|
| 344 |
+
$this->pagehook = add_options_page(
|
| 345 |
+
$this->get_setting( 'page_title' ),
|
| 346 |
+
$this->get_setting( 'menu_title' ),
|
| 347 |
+
$this->get_setting( 'capability' ),
|
| 348 |
+
$this->get_setting( 'page_slug' ),
|
| 349 |
+
array( $this, 'call_page_content' )
|
| 350 |
+
);
|
| 351 |
+
break;
|
| 352 |
+
|
| 353 |
+
}
|
| 354 |
+
|
| 355 |
+
// execute action
|
| 356 |
+
do_action( 'meta_box_load', $this );
|
| 357 |
+
|
| 358 |
+
// load page
|
| 359 |
+
add_action( 'load-' . $this->pagehook, array( $this, 'call_load_page' ) );
|
| 360 |
+
}
|
| 361 |
+
|
| 362 |
+
/**
|
| 363 |
+
* Admin head callback
|
| 364 |
+
*/
|
| 365 |
+
public function call_admin_head() {
|
| 366 |
+
?>
|
| 367 |
+
<style type="text/css">
|
| 368 |
+
.postbox-container { padding-right:1%; float:left ; /* for WP < 3.1 */ }
|
| 369 |
+
.postbox-container .meta-box-sortables { min-height:200px; }
|
| 370 |
+
.postbox-container .postbox { min-width:0; }
|
| 371 |
+
.postbox-container .postbox .inside { margin:10px 0 ; padding:0 10px; } /* for WP < 3.2 */
|
| 372 |
+
</style>
|
| 373 |
+
<?php
|
| 374 |
+
}
|
| 375 |
+
|
| 376 |
+
/**
|
| 377 |
+
* Load page callback
|
| 378 |
+
*/
|
| 379 |
+
public function call_load_page() {
|
| 380 |
+
// execute action
|
| 381 |
+
do_action( 'meta_box_load-' . $this->get_setting( 'page_slug' ), $this );
|
| 382 |
+
|
| 383 |
+
// add script for meta boxes
|
| 384 |
+
wp_enqueue_script( 'postbox' );
|
| 385 |
+
|
| 386 |
+
// add to admin head
|
| 387 |
+
add_action( 'admin_head', array( $this, 'call_admin_head' ) );
|
| 388 |
+
|
| 389 |
+
// add screen settings filter
|
| 390 |
+
add_filter( 'screen_settings', array( $this, 'call_screen_settings') );
|
| 391 |
+
|
| 392 |
+
// add help text
|
| 393 |
+
add_filter( 'contextual_help', array( $this, 'call_contextual_help' ) );
|
| 394 |
+
|
| 395 |
+
// columns
|
| 396 |
+
if ( function_exists( 'add_screen_option' ) ) {
|
| 397 |
+
$count = count( $this->get_setting( 'column_widths' ) );
|
| 398 |
+
|
| 399 |
+
add_screen_option( 'layout_columns', array(
|
| 400 |
+
'max' => $count,
|
| 401 |
+
'default' => min( $count, $this->get_setting( 'default_columns' ) )
|
| 402 |
+
));
|
| 403 |
+
}
|
| 404 |
+
|
| 405 |
+
// add meta boxes
|
| 406 |
+
$nr = 0;
|
| 407 |
+
foreach ( $this->meta_boxes AS $box ) {
|
| 408 |
+
$title = $box[ 'title' ];
|
| 409 |
+
$id = ( isset( $box[ 'id' ] ) ) ? $box[ 'id' ] : sanitize_title_with_dashes( $title .'-'. ++$nr, 'meta-box-' . $nr );
|
| 410 |
+
$callback = ( is_string( $box[ 'callback' ] ) && method_exists( $this, $box[ 'callback' ] ) ) ? array( $this, $box[ 'callback' ] ) : $box[ 'callback' ];
|
| 411 |
+
$context = $box[ 'context' ];
|
| 412 |
+
$priority = $box[ 'priority' ];
|
| 413 |
+
|
| 414 |
+
// set context
|
| 415 |
+
if ( $context == 2 OR strtolower( $context ) == 'side' ) {
|
| 416 |
+
$context = 'side';
|
| 417 |
+
} elseif ( $context == 3 OR strtolower( $context ) == 'column3' ) {
|
| 418 |
+
$context = 'column3';
|
| 419 |
+
} elseif ( $context == 4 OR strtolower( $context ) == 'column4' ) {
|
| 420 |
+
$context = 'column4';
|
| 421 |
+
} else { // default
|
| 422 |
+
$context = 'normal';
|
| 423 |
+
}
|
| 424 |
+
|
| 425 |
+
// add meta box
|
| 426 |
+
add_meta_box( $id, $title, $callback, $this->pagehook, $context, $priority ); // $callback_args, doesn't seem to work
|
| 427 |
+
}
|
| 428 |
+
}
|
| 429 |
+
|
| 430 |
+
/**
|
| 431 |
+
* Screen settings (callback)
|
| 432 |
+
* @param string $content
|
| 433 |
+
* @return string
|
| 434 |
+
*/
|
| 435 |
+
public function call_screen_settings( $content ) {
|
| 436 |
+
if ( self::get_current_screen()->id == convert_to_screen( $this->pagehook )->id ) {
|
| 437 |
+
// apply filters for this meta box page
|
| 438 |
+
$content = apply_filters( 'meta_box_screen_settings-' . $this->get_setting( 'page_slug' ), $content, $this->get_current_screen(), $this );
|
| 439 |
+
}
|
| 440 |
+
|
| 441 |
+
return $content;
|
| 442 |
+
}
|
| 443 |
+
|
| 444 |
+
/**
|
| 445 |
+
* Contextual help (callback)
|
| 446 |
+
* @param string $content
|
| 447 |
+
* @return string
|
| 448 |
+
*/
|
| 449 |
+
public function call_contextual_help( $content ) {
|
| 450 |
+
$current_screen = $this->get_current_screen();
|
| 451 |
+
|
| 452 |
+
if ( $current_screen->id == convert_to_screen( $this->pagehook )->id ) {
|
| 453 |
+
// apply filters for this meta box page
|
| 454 |
+
$content = apply_filters( 'meta_box_contextual_help-' . $this->get_setting( 'page_slug' ), $content, $current_screen->id, $current_screen, $this );
|
| 455 |
+
}
|
| 456 |
+
|
| 457 |
+
return $content;
|
| 458 |
+
}
|
| 459 |
+
|
| 460 |
+
/**
|
| 461 |
+
* Display admin page content (callback)
|
| 462 |
+
*/
|
| 463 |
+
public function call_page_content() {
|
| 464 |
+
echo '<div class="wrap">';
|
| 465 |
+
|
| 466 |
+
// page title
|
| 467 |
+
$meta_boxes_page_title = apply_filters( 'meta_box_page_title', $this->get_page_title(), $this );
|
| 468 |
+
echo apply_filters( 'meta_box_page_title-' . $this->get_setting( 'page_slug' ), $meta_boxes_page_title, $this );
|
| 469 |
+
|
| 470 |
+
// page content
|
| 471 |
+
$meta_boxes_content = apply_filters( 'meta_box_page_content', $this->get_page_content(), $this );
|
| 472 |
+
echo apply_filters( 'meta_box_page_content-' . $this->get_setting( 'page_slug' ), $meta_boxes_content, $this );
|
| 473 |
+
|
| 474 |
+
echo '</div>';
|
| 475 |
+
}
|
| 476 |
+
|
| 477 |
+
/**
|
| 478 |
+
* Get page title
|
| 479 |
+
* Can be changed by using adding filters, see add_title_filter()
|
| 480 |
+
* @return string
|
| 481 |
+
*/
|
| 482 |
+
private function get_page_title() {
|
| 483 |
+
return '<h2>' . get_admin_page_title() . '</h2>';
|
| 484 |
+
}
|
| 485 |
+
|
| 486 |
+
/**
|
| 487 |
+
* Get meta boxes content. Can be changed by adding filters, see add_content_filter()
|
| 488 |
+
* @return string
|
| 489 |
+
*/
|
| 490 |
+
private function get_page_content() {
|
| 491 |
+
return self::get_ob_callback( array( $this, 'show_meta_boxes' ) );
|
| 492 |
+
}
|
| 493 |
+
|
| 494 |
+
/**
|
| 495 |
+
* Display meta boxes content
|
| 496 |
+
*/
|
| 497 |
+
private function show_meta_boxes() {
|
| 498 |
+
$opt_column_widths = $this->get_setting( 'column_widths' );
|
| 499 |
+
$hide2 = $hide3 = $hide4 = '';
|
| 500 |
+
switch ( self::get_screen_layout_columns( $this->get_setting( 'default_columns' ) ) ) {
|
| 501 |
+
case 4:
|
| 502 |
+
$column_widths = $opt_column_widths[ 4 ];
|
| 503 |
+
break;
|
| 504 |
+
case 3:
|
| 505 |
+
$column_widths = $opt_column_widths[ 3 ];
|
| 506 |
+
$hide4 = 'display:none;';
|
| 507 |
+
break;
|
| 508 |
+
case 2:
|
| 509 |
+
$column_widths = $opt_column_widths[ 2 ];
|
| 510 |
+
$hide3 = $hide4 = 'display:none;';
|
| 511 |
+
break;
|
| 512 |
+
default:
|
| 513 |
+
$column_widths = $opt_column_widths[ 1 ];
|
| 514 |
+
$hide2 = $hide3 = $hide4 = 'display:none;';
|
| 515 |
+
}
|
| 516 |
+
|
| 517 |
+
$column_widths = array_pad( $column_widths, 4, 0 );
|
| 518 |
+
?>
|
| 519 |
+
<div id='<?php echo $this->pagehook ?>-widgets' class='metabox-holder'>
|
| 520 |
+
<div class='postbox-container' style='width:<?php echo $column_widths[0] ?>%'>
|
| 521 |
+
<?php do_meta_boxes( $this->pagehook, 'normal', '' ); ?>
|
| 522 |
+
</div>
|
| 523 |
+
|
| 524 |
+
<div class='postbox-container' style='<?php echo $hide2 ?>width:<?php echo $column_widths[1] ?>%'>
|
| 525 |
+
<?php do_meta_boxes( $this->pagehook, 'side', '' ); ?>
|
| 526 |
+
</div>
|
| 527 |
+
|
| 528 |
+
<div class='postbox-container' style='<?php echo $hide3 ?>width:<?php echo $column_widths[2] ?>%'>
|
| 529 |
+
<?php do_meta_boxes( $this->pagehook, 'column3', '' ); ?>
|
| 530 |
+
</div>
|
| 531 |
+
|
| 532 |
+
<div class='postbox-container' style='<?php echo $hide4 ?>width:<?php echo $column_widths[3] ?>%'>
|
| 533 |
+
<?php do_meta_boxes( $this->pagehook, 'column4', '' ); ?>
|
| 534 |
+
</div>
|
| 535 |
+
</div>
|
| 536 |
+
|
| 537 |
+
<form style="display:none" method="get" action="">
|
| 538 |
+
<?php wp_nonce_field( 'closedpostboxes', 'closedpostboxesnonce', false ); ?>
|
| 539 |
+
<?php wp_nonce_field( 'meta-box-order', 'meta-box-order-nonce', false ); ?>
|
| 540 |
+
</form>
|
| 541 |
+
|
| 542 |
+
<script type="text/javascript">
|
| 543 |
+
//<![CDATA[
|
| 544 |
+
jQuery( document ).ready( function( $ ){
|
| 545 |
+
var columnWidths = <?php echo json_encode( $opt_column_widths ) ?>,
|
| 546 |
+
$boxes = $( '.postbox-container' ),
|
| 547 |
+
setColumnWidths = function () {
|
| 548 |
+
var c = $( 'input[name="screen_columns"]:checked' ).val();
|
| 549 |
+
|
| 550 |
+
// first hide all boxes
|
| 551 |
+
$boxes.hide();
|
| 552 |
+
|
| 553 |
+
// set width and show boxes
|
| 554 |
+
for ( var x = 0; x < columnWidths[ c ].length; x++ ) {
|
| 555 |
+
$boxes.eq( x )
|
| 556 |
+
.css( 'width', columnWidths[ c ][ x ]+ '%' )
|
| 557 |
+
.show();
|
| 558 |
+
}
|
| 559 |
+
};
|
| 560 |
+
|
| 561 |
+
// radio screen columns
|
| 562 |
+
$( 'input[name="screen_columns"]' )
|
| 563 |
+
.click(function(){
|
| 564 |
+
if ( $( 'input[name="screen_columns"]:checked' ).val() == $( this ).val() ) {
|
| 565 |
+
setTimeout(function(){
|
| 566 |
+
setColumnWidths();
|
| 567 |
+
}, 1 );
|
| 568 |
+
}
|
| 569 |
+
})
|
| 570 |
+
.change(function( e ){
|
| 571 |
+
setColumnWidths();
|
| 572 |
+
|
| 573 |
+
// prevent
|
| 574 |
+
e.stopImmediatePropagation();
|
| 575 |
+
});
|
| 576 |
+
|
| 577 |
+
// trigger change event of selected column
|
| 578 |
+
$( 'input[name="screen_columns"]:checked' ).change();
|
| 579 |
+
|
| 580 |
+
<?php if ( self::wp_version( '3.2', '<' ) ): ?>
|
| 581 |
+
// for WP < 3.2
|
| 582 |
+
|
| 583 |
+
// close postboxes that should be closed
|
| 584 |
+
$( '.if-js-closed' ).removeClass( 'if-js-closed' ).addClass( 'closed' );
|
| 585 |
+
|
| 586 |
+
// Loading saved screen settings
|
| 587 |
+
postboxes.add_postbox_toggles( '<?php echo $this->pagehook ?>' );
|
| 588 |
+
<?php endif; ?>
|
| 589 |
+
});
|
| 590 |
+
//]]>
|
| 591 |
+
</script>
|
| 592 |
+
<?php
|
| 593 |
+
}
|
| 594 |
+
|
| 595 |
+
/**
|
| 596 |
+
* Static helpers
|
| 597 |
+
*/
|
| 598 |
+
|
| 599 |
+
/**
|
| 600 |
+
* Get content displayed by given callback
|
| 601 |
+
* @param mixed $callback
|
| 602 |
+
* @return string
|
| 603 |
+
* @static
|
| 604 |
+
*/
|
| 605 |
+
public static function get_ob_callback( $callback ) {
|
| 606 |
+
// start output buffer
|
| 607 |
+
ob_start();
|
| 608 |
+
|
| 609 |
+
// call callback
|
| 610 |
+
call_user_func( $callback );
|
| 611 |
+
|
| 612 |
+
// get the view content
|
| 613 |
+
$content = ob_get_contents();
|
| 614 |
+
|
| 615 |
+
// clean output buffer
|
| 616 |
+
ob_end_clean();
|
| 617 |
+
|
| 618 |
+
return $content;
|
| 619 |
+
}
|
| 620 |
+
|
| 621 |
+
/**
|
| 622 |
+
* Get current version of WP or compare versions
|
| 623 |
+
* @global string $wp_version
|
| 624 |
+
* @return mixed
|
| 625 |
+
* @static
|
| 626 |
+
*/
|
| 627 |
+
public static function wp_version( $compare_version = NULL, $operator = NULL ) {
|
| 628 |
+
global $wp_version;
|
| 629 |
+
$cur_wp_version = preg_replace( '/-.*$/', '', $wp_version );
|
| 630 |
+
|
| 631 |
+
if ( $compare_version === NULL )
|
| 632 |
+
return $cur_wp_version;
|
| 633 |
+
|
| 634 |
+
// check comparison
|
| 635 |
+
return version_compare( $cur_wp_version, $compare_version, $operator );
|
| 636 |
+
}
|
| 637 |
+
|
| 638 |
+
/**
|
| 639 |
+
* Return global WP value of $screen_layout_columns
|
| 640 |
+
* @global integer $screen_layout_columns
|
| 641 |
+
* @return integer
|
| 642 |
+
* @static
|
| 643 |
+
*/
|
| 644 |
+
public static function get_screen_layout_columns( $default = NULL ) {
|
| 645 |
+
global $screen_layout_columns;
|
| 646 |
+
return ( empty( $screen_layout_columns ) ) ? $default : $screen_layout_columns;
|
| 647 |
+
}
|
| 648 |
+
|
| 649 |
+
/**
|
| 650 |
+
* Return global WP value of $current_screen
|
| 651 |
+
* @global string $current_screen
|
| 652 |
+
* @return string
|
| 653 |
+
* @static
|
| 654 |
+
*/
|
| 655 |
+
public static function get_current_screen( $default = NULL ) {
|
| 656 |
+
global $current_screen;
|
| 657 |
+
return $current_screen;
|
| 658 |
+
}
|
| 659 |
+
|
| 660 |
+
} // End WP_Meta_Box_Page_01 Class
|
| 661 |
+
|
| 662 |
+
endif;
|
| 663 |
+
|
| 664 |
/* ommit PHP closing tag, to prevent unwanted whitespace at the end of the parts generated by the included files */
|
includes/wp-plugin-dev-classes/class-wp-option-forms.php
CHANGED
|
@@ -1,360 +1,360 @@
|
|
| 1 |
-
<?php defined( 'ABSPATH' ) OR die( 'No direct access.' );
|
| 2 |
-
if ( ! class_exists( 'WP_Option_Forms_01' ) ):
|
| 3 |
-
/**
|
| 4 |
-
* WP_Option_Forms_01 Class
|
| 5 |
-
*
|
| 6 |
-
* Simple class for creating option forms of a the same option_group.
|
| 7 |
-
* Also with Ajax save support.
|
| 8 |
-
*
|
| 9 |
-
* Requires WordPress 3.0+ and PHP 5.2+
|
| 10 |
-
*
|
| 11 |
-
* @version 0.1
|
| 12 |
-
* @author Victor Villaverde Laan
|
| 13 |
-
* @link http://www.freelancephp.net/
|
| 14 |
-
* @license Dual licensed under the MIT and GPL licenses
|
| 15 |
-
*/
|
| 16 |
-
class WP_Option_Forms_01 {
|
| 17 |
-
|
| 18 |
-
/**
|
| 19 |
-
* Name used as prefix for saving option names
|
| 20 |
-
* @var string
|
| 21 |
-
*/
|
| 22 |
-
protected $name = NULL;
|
| 23 |
-
|
| 24 |
-
/**
|
| 25 |
-
* Option names and values
|
| 26 |
-
* @var string
|
| 27 |
-
*/
|
| 28 |
-
protected $options = array();
|
| 29 |
-
|
| 30 |
-
/**
|
| 31 |
-
* Current option name
|
| 32 |
-
* @var string
|
| 33 |
-
*/
|
| 34 |
-
protected $current_option = NULL;
|
| 35 |
-
|
| 36 |
-
|
| 37 |
-
/**
|
| 38 |
-
* Constructor
|
| 39 |
-
* @param array $name
|
| 40 |
-
* @param array $options Optional
|
| 41 |
-
*/
|
| 42 |
-
public function __construct( $name, $options = array() ) {
|
| 43 |
-
$this->name = sanitize_title_with_dashes( $name );
|
| 44 |
-
|
| 45 |
-
// set option names
|
| 46 |
-
foreach ( $options AS $option_name => $values ) {
|
| 47 |
-
$this->add_option( $option_name, $values );
|
| 48 |
-
}
|
| 49 |
-
|
| 50 |
-
// actions
|
| 51 |
-
add_action( 'wp_ajax_wpof_update_options', array( $this, 'call_wp_ajax' ) );
|
| 52 |
-
add_action( 'admin_menu', array( $this, 'call_admin_menu' ) );
|
| 53 |
-
}
|
| 54 |
-
|
| 55 |
-
/**
|
| 56 |
-
* Admin menu callback
|
| 57 |
-
*/
|
| 58 |
-
public function call_admin_menu() {
|
| 59 |
-
// Register settings
|
| 60 |
-
foreach ( $this->options AS $option_name => $values ) {
|
| 61 |
-
register_setting( $option_name, $option_name );
|
| 62 |
-
}
|
| 63 |
-
|
| 64 |
-
// script
|
| 65 |
-
wp_enqueue_script( 'option-forms', plugins_url( 'js/wp-option-forms.js', WP_EXTERNAL_LINKS_FILE ), array( 'jquery' ), '1.0' );
|
| 66 |
-
}
|
| 67 |
-
|
| 68 |
-
/**
|
| 69 |
-
* Ajax call for saving option values
|
| 70 |
-
*/
|
| 71 |
-
public function call_wp_ajax() {
|
| 72 |
-
check_ajax_referer( 'wpof_update_options', 'wpof-nonce' );
|
| 73 |
-
|
| 74 |
-
$option_name = $_POST[ 'ajax_option_name' ];
|
| 75 |
-
$value = NULL;
|
| 76 |
-
|
| 77 |
-
if ( isset( $_POST[ $option_name ] ) )
|
| 78 |
-
$value = $_POST[ $option_name ];
|
| 79 |
-
|
| 80 |
-
if ( ! is_array( $value ) )
|
| 81 |
-
$value = trim( $value );
|
| 82 |
-
|
| 83 |
-
$value = stripslashes_deep( $value );
|
| 84 |
-
|
| 85 |
-
update_option( $option_name, $value );
|
| 86 |
-
|
| 87 |
-
die( '1' );
|
| 88 |
-
}
|
| 89 |
-
|
| 90 |
-
/**
|
| 91 |
-
* Add option (or reset option when already exists)
|
| 92 |
-
* @param string $option_name
|
| 93 |
-
* @param array $default_values Optional
|
| 94 |
-
* @return this
|
| 95 |
-
*/
|
| 96 |
-
public function add_option( $option_name, $default_values = array() ) {
|
| 97 |
-
// set values
|
| 98 |
-
$saved_values = get_option( $this->name .'-'. $option_name );
|
| 99 |
-
|
| 100 |
-
if ( empty( $saved_values ) ) {
|
| 101 |
-
foreach ( $default_values AS $key => $value )
|
| 102 |
-
$values[ $key ] = $value;
|
| 103 |
-
} else {
|
| 104 |
-
foreach ( $default_values AS $key => $value )
|
| 105 |
-
$values[ $key ] = '';
|
| 106 |
-
|
| 107 |
-
foreach ( $saved_values AS $key => $value )
|
| 108 |
-
$values[ $key ] = $value;
|
| 109 |
-
}
|
| 110 |
-
|
| 111 |
-
// option and values
|
| 112 |
-
$this->options[ $this->name .'-'. $option_name ] = $values;
|
| 113 |
-
return $this;
|
| 114 |
-
}
|
| 115 |
-
|
| 116 |
-
/**
|
| 117 |
-
* Set current option to use
|
| 118 |
-
* @param string $option_name
|
| 119 |
-
* @return this
|
| 120 |
-
*/
|
| 121 |
-
public function set_current_option( $option_name ) {
|
| 122 |
-
$this->current_option = $this->name .'-'. $option_name;
|
| 123 |
-
return $this;
|
| 124 |
-
}
|
| 125 |
-
|
| 126 |
-
/**
|
| 127 |
-
* Get opening form with all nescessary WP fields
|
| 128 |
-
* @param boolean $ajaxSave Optional
|
| 129 |
-
* @param array $attrs Optional
|
| 130 |
-
* @return string
|
| 131 |
-
*/
|
| 132 |
-
public function open_form( $ajaxSave = TRUE, $attrs = array() ) {
|
| 133 |
-
// set class for ajax or non-ajax form
|
| 134 |
-
$attrs[ 'class' ] = ( ( $ajaxSave ) ? 'ajax-form' : 'no-ajax-form' )
|
| 135 |
-
. ( ( key_exists( 'class', $attrs ) ) ? ' '. $attrs[ 'class' ] : '' );
|
| 136 |
-
|
| 137 |
-
// show start form
|
| 138 |
-
$html = '';
|
| 139 |
-
$html .= '<form method="post" action="options.php" '. $this->attrs( $attrs ) .'>';
|
| 140 |
-
|
| 141 |
-
if ( $ajaxSave ) {
|
| 142 |
-
$html .= wp_nonce_field( 'wpof_update_options', 'wpof-nonce', FALSE, FALSE );
|
| 143 |
-
$html .= '<input type="hidden" name="action" value="wpof_update_options" />';
|
| 144 |
-
$html .= '<input type="hidden" name="ajax_option_name" value="'. $this->current_option .'" />';
|
| 145 |
-
|
| 146 |
-
// instead of using settings_fields();
|
| 147 |
-
$html .= '<input type="hidden" name="option_page" value="' . esc_attr( $this->current_option ) . '" />';
|
| 148 |
-
$html .= wp_nonce_field( $this->current_option . '-options', '_wpnonce', TRUE, FALSE );
|
| 149 |
-
} else {
|
| 150 |
-
// instead of using settings_fields();
|
| 151 |
-
$html .= '<input type="hidden" name="option_page" value="' . esc_attr( $this->current_option ) . '" />';
|
| 152 |
-
$html .= '<input type="hidden" name="action" value="update" />';
|
| 153 |
-
$html .= wp_nonce_field( $this->current_option . '-options', '_wpnonce', TRUE, FALSE );
|
| 154 |
-
}
|
| 155 |
-
|
| 156 |
-
return $html;
|
| 157 |
-
}
|
| 158 |
-
|
| 159 |
-
/**
|
| 160 |
-
* Get script for saving screen option
|
| 161 |
-
* @param string $option_name
|
| 162 |
-
* @param string $key
|
| 163 |
-
* @return string
|
| 164 |
-
*/
|
| 165 |
-
public function open_screen_option( $option_name, $key ) {
|
| 166 |
-
$this->set_current_option( $option_name );
|
| 167 |
-
|
| 168 |
-
$html = '';
|
| 169 |
-
$html .= '<script type="text/javascript">' . "\n";
|
| 170 |
-
$html .= '//<![CDATA[' . "\n";
|
| 171 |
-
$html .= 'jQuery( document ).ready( function( $ ){' . "\n";
|
| 172 |
-
$html .= "\t" . '// save screen option' . "\n";
|
| 173 |
-
$html .= "\t" . '$( "#screen-meta #'. $key .'" )' . "\n";
|
| 174 |
-
$html .= "\t\t" . '.change(function(){' . "\n";
|
| 175 |
-
$html .= "\t\t\t" . 'var self = this;' . "\n";
|
| 176 |
-
$html .= "\t\t\t" . '$.post( ajaxurl, {' . "\n";
|
| 177 |
-
$html .= "\t\t\t\t" . 'action: "wpof_update_options",' . "\n";
|
| 178 |
-
$html .= "\t\t\t\t" . '"wpof-nonce": "'. wp_create_nonce( 'wpof_update_options' ) .'",' . "\n";
|
| 179 |
-
$html .= "\t\t\t\t" . 'ajax_option_name: "'. $this->current_option .'",' . "\n";
|
| 180 |
-
$html .= "\t\t\t\t" . '"'. $this->field_name( $key ) .'": $( this ).val()' . "\n";
|
| 181 |
-
$html .= "\t\t\t" . '}, function () {' . "\n";
|
| 182 |
-
$html .= "\t\t\t\t" . '$( self ).trigger( "ajax_updated" );' . "\n";
|
| 183 |
-
$html .= "\t\t\t" . '});' . "\n";
|
| 184 |
-
$html .= "\t\t" . '});' . "\n";
|
| 185 |
-
$html .= '});' . "\n";
|
| 186 |
-
$html .= '//]]>' . "\n";
|
| 187 |
-
$html .= '</script>' . "\n";
|
| 188 |
-
|
| 189 |
-
return $html;
|
| 190 |
-
}
|
| 191 |
-
|
| 192 |
-
/**
|
| 193 |
-
* Get closing form
|
| 194 |
-
* @return string
|
| 195 |
-
*/
|
| 196 |
-
public function close_form() {
|
| 197 |
-
return '</form>';
|
| 198 |
-
}
|
| 199 |
-
|
| 200 |
-
/**
|
| 201 |
-
* Text field
|
| 202 |
-
* @param string $key
|
| 203 |
-
* @param array $attrs Optional
|
| 204 |
-
* @return string
|
| 205 |
-
*/
|
| 206 |
-
public function text( $key, $attrs = array() ) {
|
| 207 |
-
if ( ! key_exists( 'class', $attrs ) )
|
| 208 |
-
$attrs[ 'class' ] = 'regular-text';
|
| 209 |
-
|
| 210 |
-
return '<input type="text" '. $this->attrs( $attrs, $key, $this->value( $key ) ) .' />';
|
| 211 |
-
}
|
| 212 |
-
|
| 213 |
-
/**
|
| 214 |
-
* Text field
|
| 215 |
-
* @param string $key
|
| 216 |
-
* @param array $attrs Optional
|
| 217 |
-
* @return string
|
| 218 |
-
*/
|
| 219 |
-
public function textarea( $key, $attrs = array() ) {
|
| 220 |
-
if ( ! key_exists( 'class', $attrs ) )
|
| 221 |
-
$attrs[ 'class' ] = 'large-text';
|
| 222 |
-
|
| 223 |
-
return '<textarea '. $this->attrs( $attrs, $key ) .'>'. $this->value( $key ) .'</textarea>';
|
| 224 |
-
}
|
| 225 |
-
|
| 226 |
-
/**
|
| 227 |
-
* Radio field
|
| 228 |
-
* @param string $key
|
| 229 |
-
* @param mixed $value
|
| 230 |
-
* @param array $attrs Optional
|
| 231 |
-
* @return string
|
| 232 |
-
*/
|
| 233 |
-
public function radio( $key, $value, $attrs = array() ) {
|
| 234 |
-
$checked = ( $value == $this->value( $key ) ) ? ' checked="checked"' : '';
|
| 235 |
-
return '<input type="radio" '. $this->attrs( $attrs, $key, $value )
|
| 236 |
-
. $checked . ' />';
|
| 237 |
-
}
|
| 238 |
-
|
| 239 |
-
/**
|
| 240 |
-
* Checkbox field
|
| 241 |
-
* @param string $key
|
| 242 |
-
* @param mixed $value
|
| 243 |
-
* @param array $attrs Optional
|
| 244 |
-
* @return string
|
| 245 |
-
*/
|
| 246 |
-
public function checkbox( $key, $value, $attrs = array() ) {
|
| 247 |
-
$checked = ( $value == $this->value( $key ) ) ? ' checked="checked"' : '';
|
| 248 |
-
return '<input type="checkbox" '. $this->attrs( $attrs, $key, $value )
|
| 249 |
-
. $checked . ' />';
|
| 250 |
-
}
|
| 251 |
-
|
| 252 |
-
/**
|
| 253 |
-
* Select field
|
| 254 |
-
* @param string $key
|
| 255 |
-
* @param array $options Optional
|
| 256 |
-
* @param array $attrs Optional
|
| 257 |
-
* @return string
|
| 258 |
-
*/
|
| 259 |
-
public function select( $key, $options = array(), $attrs = array() ) {
|
| 260 |
-
$html = '<select '. $this->attrs( $attrs, $key ) .'>';
|
| 261 |
-
|
| 262 |
-
foreach ( $options AS $value => $label ) {
|
| 263 |
-
$selected = ( $value == $this->value( $key ) ) ? ' selected="selected"' : '';
|
| 264 |
-
$html .= '<option value="'. $value .'"'. $selected .'>'. $label .'</option>';
|
| 265 |
-
}
|
| 266 |
-
|
| 267 |
-
$html .= '</select>';
|
| 268 |
-
return $html;
|
| 269 |
-
}
|
| 270 |
-
|
| 271 |
-
/**
|
| 272 |
-
* Submit button
|
| 273 |
-
* @param array $attrs Optional
|
| 274 |
-
* @return string
|
| 275 |
-
*/
|
| 276 |
-
public function submit( $attrs = array() ) {
|
| 277 |
-
// set class attr
|
| 278 |
-
$attrs[ 'class' ] = 'button-primary'. ( ( key_exists( 'class', $attrs ) ) ? ' '. $attrs[ 'class' ] : '' );
|
| 279 |
-
|
| 280 |
-
// show submit
|
| 281 |
-
$html = '';
|
| 282 |
-
$html .= '<p class="button-controls" style="text-align:right;">';
|
| 283 |
-
$html .= '<img alt="" title="" class="ajax-feedback" src="'. get_bloginfo( 'url' ) .'/wp-admin/images/wpspin_light.gif" style="visibility: hidden;" />';
|
| 284 |
-
$html .= '<input type="submit" '. $this->attrs( $attrs, '', __( 'Save Changes' ) ) .' />';
|
| 285 |
-
$html .= '</p>';
|
| 286 |
-
return $html;
|
| 287 |
-
}
|
| 288 |
-
|
| 289 |
-
/**
|
| 290 |
-
* Get field name of given key
|
| 291 |
-
* @param string $key
|
| 292 |
-
* @return string
|
| 293 |
-
*/
|
| 294 |
-
public function field_name( $key ) {
|
| 295 |
-
return $this->current_option . '[' . $key . ']';
|
| 296 |
-
}
|
| 297 |
-
|
| 298 |
-
/**
|
| 299 |
-
* Get value of given option key
|
| 300 |
-
* @param string $key
|
| 301 |
-
* @param mixed $default_value Optional
|
| 302 |
-
* @param boolean $option_name Optional, search in given option_name instead of the current option
|
| 303 |
-
* @return mixed
|
| 304 |
-
*/
|
| 305 |
-
public function value( $key, $default_value = NULL, $option_name = NULL ) {
|
| 306 |
-
if ( $option_name === NULL ) {
|
| 307 |
-
$option = $this->current_option;
|
| 308 |
-
} else {
|
| 309 |
-
$option = $this->name . '-' . $option_name;
|
| 310 |
-
}
|
| 311 |
-
|
| 312 |
-
if (!isset($this->options[ $option ])) {
|
| 313 |
-
return $default_value;
|
| 314 |
-
}
|
| 315 |
-
|
| 316 |
-
$values = $this->options[ $option ];
|
| 317 |
-
|
| 318 |
-
return ( is_array( $values ) AND key_exists( $key, $values ) AND $values[ $key ] !== NULL ) ? $values[ $key ] : $default_value;
|
| 319 |
-
}
|
| 320 |
-
|
| 321 |
-
/**
|
| 322 |
-
* Delete and unregister option
|
| 323 |
-
*/
|
| 324 |
-
public function delete_options() {
|
| 325 |
-
foreach ( $this->options AS $option_name => $values ) {
|
| 326 |
-
delete_option( $option_name );
|
| 327 |
-
}
|
| 328 |
-
}
|
| 329 |
-
|
| 330 |
-
/**
|
| 331 |
-
* Get string of given attributes
|
| 332 |
-
* @param array $attrs
|
| 333 |
-
* @param string $key Optional
|
| 334 |
-
* @param mixed $value Optional
|
| 335 |
-
* @return string
|
| 336 |
-
*/
|
| 337 |
-
protected function attrs( $attrs, $key = NULL, $value = NULL ) {
|
| 338 |
-
$str = '';
|
| 339 |
-
|
| 340 |
-
// set name, id, value attr
|
| 341 |
-
if ( $key !== NULL ) {
|
| 342 |
-
$str .= 'name="' . $this->field_name( $key ) .'" ';
|
| 343 |
-
if ( ! key_exists( 'id', $attrs ) )
|
| 344 |
-
$str .= 'id="' . $key .'" ';
|
| 345 |
-
}
|
| 346 |
-
|
| 347 |
-
if ( $value !== NULL )
|
| 348 |
-
$str .= 'value="' . $value .'" ';
|
| 349 |
-
|
| 350 |
-
foreach ( $attrs AS $attr => $value )
|
| 351 |
-
$str .= $attr .'="'. $value .'" ';
|
| 352 |
-
|
| 353 |
-
return $str;
|
| 354 |
-
}
|
| 355 |
-
|
| 356 |
-
} // End WP_Option_Forms_01
|
| 357 |
-
|
| 358 |
-
endif;
|
| 359 |
-
|
| 360 |
/* ommit PHP closing tag, to prevent unwanted whitespace at the end of the parts generated by the included files */
|
| 1 |
+
<?php defined( 'ABSPATH' ) OR die( 'No direct access.' );
|
| 2 |
+
if ( ! class_exists( 'WP_Option_Forms_01' ) ):
|
| 3 |
+
/**
|
| 4 |
+
* WP_Option_Forms_01 Class
|
| 5 |
+
*
|
| 6 |
+
* Simple class for creating option forms of a the same option_group.
|
| 7 |
+
* Also with Ajax save support.
|
| 8 |
+
*
|
| 9 |
+
* Requires WordPress 3.0+ and PHP 5.2+
|
| 10 |
+
*
|
| 11 |
+
* @version 0.1
|
| 12 |
+
* @author Victor Villaverde Laan
|
| 13 |
+
* @link http://www.freelancephp.net/
|
| 14 |
+
* @license Dual licensed under the MIT and GPL licenses
|
| 15 |
+
*/
|
| 16 |
+
class WP_Option_Forms_01 {
|
| 17 |
+
|
| 18 |
+
/**
|
| 19 |
+
* Name used as prefix for saving option names
|
| 20 |
+
* @var string
|
| 21 |
+
*/
|
| 22 |
+
protected $name = NULL;
|
| 23 |
+
|
| 24 |
+
/**
|
| 25 |
+
* Option names and values
|
| 26 |
+
* @var string
|
| 27 |
+
*/
|
| 28 |
+
protected $options = array();
|
| 29 |
+
|
| 30 |
+
/**
|
| 31 |
+
* Current option name
|
| 32 |
+
* @var string
|
| 33 |
+
*/
|
| 34 |
+
protected $current_option = NULL;
|
| 35 |
+
|
| 36 |
+
|
| 37 |
+
/**
|
| 38 |
+
* Constructor
|
| 39 |
+
* @param array $name
|
| 40 |
+
* @param array $options Optional
|
| 41 |
+
*/
|
| 42 |
+
public function __construct( $name, $options = array() ) {
|
| 43 |
+
$this->name = sanitize_title_with_dashes( $name );
|
| 44 |
+
|
| 45 |
+
// set option names
|
| 46 |
+
foreach ( $options AS $option_name => $values ) {
|
| 47 |
+
$this->add_option( $option_name, $values );
|
| 48 |
+
}
|
| 49 |
+
|
| 50 |
+
// actions
|
| 51 |
+
add_action( 'wp_ajax_wpof_update_options', array( $this, 'call_wp_ajax' ) );
|
| 52 |
+
add_action( 'admin_menu', array( $this, 'call_admin_menu' ) );
|
| 53 |
+
}
|
| 54 |
+
|
| 55 |
+
/**
|
| 56 |
+
* Admin menu callback
|
| 57 |
+
*/
|
| 58 |
+
public function call_admin_menu() {
|
| 59 |
+
// Register settings
|
| 60 |
+
foreach ( $this->options AS $option_name => $values ) {
|
| 61 |
+
register_setting( $option_name, $option_name );
|
| 62 |
+
}
|
| 63 |
+
|
| 64 |
+
// script
|
| 65 |
+
// wp_enqueue_script( 'option-forms', plugins_url( 'js/wp-option-forms.js', WP_EXTERNAL_LINKS_FILE ), array( 'jquery' ), '1.0' );
|
| 66 |
+
}
|
| 67 |
+
|
| 68 |
+
/**
|
| 69 |
+
* Ajax call for saving option values
|
| 70 |
+
*/
|
| 71 |
+
public function call_wp_ajax() {
|
| 72 |
+
check_ajax_referer( 'wpof_update_options', 'wpof-nonce' );
|
| 73 |
+
|
| 74 |
+
$option_name = $_POST[ 'ajax_option_name' ];
|
| 75 |
+
$value = NULL;
|
| 76 |
+
|
| 77 |
+
if ( isset( $_POST[ $option_name ] ) )
|
| 78 |
+
$value = $_POST[ $option_name ];
|
| 79 |
+
|
| 80 |
+
if ( ! is_array( $value ) )
|
| 81 |
+
$value = trim( $value );
|
| 82 |
+
|
| 83 |
+
$value = stripslashes_deep( $value );
|
| 84 |
+
|
| 85 |
+
update_option( $option_name, $value );
|
| 86 |
+
|
| 87 |
+
die( '1' );
|
| 88 |
+
}
|
| 89 |
+
|
| 90 |
+
/**
|
| 91 |
+
* Add option (or reset option when already exists)
|
| 92 |
+
* @param string $option_name
|
| 93 |
+
* @param array $default_values Optional
|
| 94 |
+
* @return this
|
| 95 |
+
*/
|
| 96 |
+
public function add_option( $option_name, $default_values = array() ) {
|
| 97 |
+
// set values
|
| 98 |
+
$saved_values = get_option( $this->name .'-'. $option_name );
|
| 99 |
+
|
| 100 |
+
if ( empty( $saved_values ) ) {
|
| 101 |
+
foreach ( $default_values AS $key => $value )
|
| 102 |
+
$values[ $key ] = $value;
|
| 103 |
+
} else {
|
| 104 |
+
foreach ( $default_values AS $key => $value )
|
| 105 |
+
$values[ $key ] = '';
|
| 106 |
+
|
| 107 |
+
foreach ( $saved_values AS $key => $value )
|
| 108 |
+
$values[ $key ] = $value;
|
| 109 |
+
}
|
| 110 |
+
|
| 111 |
+
// option and values
|
| 112 |
+
$this->options[ $this->name .'-'. $option_name ] = $values;
|
| 113 |
+
return $this;
|
| 114 |
+
}
|
| 115 |
+
|
| 116 |
+
/**
|
| 117 |
+
* Set current option to use
|
| 118 |
+
* @param string $option_name
|
| 119 |
+
* @return this
|
| 120 |
+
*/
|
| 121 |
+
public function set_current_option( $option_name ) {
|
| 122 |
+
$this->current_option = $this->name .'-'. $option_name;
|
| 123 |
+
return $this;
|
| 124 |
+
}
|
| 125 |
+
|
| 126 |
+
/**
|
| 127 |
+
* Get opening form with all nescessary WP fields
|
| 128 |
+
* @param boolean $ajaxSave Optional
|
| 129 |
+
* @param array $attrs Optional
|
| 130 |
+
* @return string
|
| 131 |
+
*/
|
| 132 |
+
public function open_form( $ajaxSave = TRUE, $attrs = array() ) {
|
| 133 |
+
// set class for ajax or non-ajax form
|
| 134 |
+
$attrs[ 'class' ] = ( ( $ajaxSave ) ? 'ajax-form' : 'no-ajax-form' )
|
| 135 |
+
. ( ( key_exists( 'class', $attrs ) ) ? ' '. $attrs[ 'class' ] : '' );
|
| 136 |
+
|
| 137 |
+
// show start form
|
| 138 |
+
$html = '';
|
| 139 |
+
$html .= '<form method="post" action="options.php" '. $this->attrs( $attrs ) .'>';
|
| 140 |
+
|
| 141 |
+
if ( $ajaxSave ) {
|
| 142 |
+
$html .= wp_nonce_field( 'wpof_update_options', 'wpof-nonce', FALSE, FALSE );
|
| 143 |
+
$html .= '<input type="hidden" name="action" value="wpof_update_options" />';
|
| 144 |
+
$html .= '<input type="hidden" name="ajax_option_name" value="'. $this->current_option .'" />';
|
| 145 |
+
|
| 146 |
+
// instead of using settings_fields();
|
| 147 |
+
$html .= '<input type="hidden" name="option_page" value="' . esc_attr( $this->current_option ) . '" />';
|
| 148 |
+
$html .= wp_nonce_field( $this->current_option . '-options', '_wpnonce', TRUE, FALSE );
|
| 149 |
+
} else {
|
| 150 |
+
// instead of using settings_fields();
|
| 151 |
+
$html .= '<input type="hidden" name="option_page" value="' . esc_attr( $this->current_option ) . '" />';
|
| 152 |
+
$html .= '<input type="hidden" name="action" value="update" />';
|
| 153 |
+
$html .= wp_nonce_field( $this->current_option . '-options', '_wpnonce', TRUE, FALSE );
|
| 154 |
+
}
|
| 155 |
+
|
| 156 |
+
return $html;
|
| 157 |
+
}
|
| 158 |
+
|
| 159 |
+
/**
|
| 160 |
+
* Get script for saving screen option
|
| 161 |
+
* @param string $option_name
|
| 162 |
+
* @param string $key
|
| 163 |
+
* @return string
|
| 164 |
+
*/
|
| 165 |
+
public function open_screen_option( $option_name, $key ) {
|
| 166 |
+
$this->set_current_option( $option_name );
|
| 167 |
+
|
| 168 |
+
$html = '';
|
| 169 |
+
$html .= '<script type="text/javascript">' . "\n";
|
| 170 |
+
$html .= '//<![CDATA[' . "\n";
|
| 171 |
+
$html .= 'jQuery( document ).ready( function( $ ){' . "\n";
|
| 172 |
+
$html .= "\t" . '// save screen option' . "\n";
|
| 173 |
+
$html .= "\t" . '$( "#screen-meta #'. $key .'" )' . "\n";
|
| 174 |
+
$html .= "\t\t" . '.change(function(){' . "\n";
|
| 175 |
+
$html .= "\t\t\t" . 'var self = this;' . "\n";
|
| 176 |
+
$html .= "\t\t\t" . '$.post( ajaxurl, {' . "\n";
|
| 177 |
+
$html .= "\t\t\t\t" . 'action: "wpof_update_options",' . "\n";
|
| 178 |
+
$html .= "\t\t\t\t" . '"wpof-nonce": "'. wp_create_nonce( 'wpof_update_options' ) .'",' . "\n";
|
| 179 |
+
$html .= "\t\t\t\t" . 'ajax_option_name: "'. $this->current_option .'",' . "\n";
|
| 180 |
+
$html .= "\t\t\t\t" . '"'. $this->field_name( $key ) .'": $( this ).val()' . "\n";
|
| 181 |
+
$html .= "\t\t\t" . '}, function () {' . "\n";
|
| 182 |
+
$html .= "\t\t\t\t" . '$( self ).trigger( "ajax_updated" );' . "\n";
|
| 183 |
+
$html .= "\t\t\t" . '});' . "\n";
|
| 184 |
+
$html .= "\t\t" . '});' . "\n";
|
| 185 |
+
$html .= '});' . "\n";
|
| 186 |
+
$html .= '//]]>' . "\n";
|
| 187 |
+
$html .= '</script>' . "\n";
|
| 188 |
+
|
| 189 |
+
return $html;
|
| 190 |
+
}
|
| 191 |
+
|
| 192 |
+
/**
|
| 193 |
+
* Get closing form
|
| 194 |
+
* @return string
|
| 195 |
+
*/
|
| 196 |
+
public function close_form() {
|
| 197 |
+
return '</form>';
|
| 198 |
+
}
|
| 199 |
+
|
| 200 |
+
/**
|
| 201 |
+
* Text field
|
| 202 |
+
* @param string $key
|
| 203 |
+
* @param array $attrs Optional
|
| 204 |
+
* @return string
|
| 205 |
+
*/
|
| 206 |
+
public function text( $key, $attrs = array() ) {
|
| 207 |
+
if ( ! key_exists( 'class', $attrs ) )
|
| 208 |
+
$attrs[ 'class' ] = 'regular-text';
|
| 209 |
+
|
| 210 |
+
return '<input type="text" '. $this->attrs( $attrs, $key, $this->value( $key ) ) .' />';
|
| 211 |
+
}
|
| 212 |
+
|
| 213 |
+
/**
|
| 214 |
+
* Text field
|
| 215 |
+
* @param string $key
|
| 216 |
+
* @param array $attrs Optional
|
| 217 |
+
* @return string
|
| 218 |
+
*/
|
| 219 |
+
public function textarea( $key, $attrs = array() ) {
|
| 220 |
+
if ( ! key_exists( 'class', $attrs ) )
|
| 221 |
+
$attrs[ 'class' ] = 'large-text';
|
| 222 |
+
|
| 223 |
+
return '<textarea '. $this->attrs( $attrs, $key ) .'>'. $this->value( $key ) .'</textarea>';
|
| 224 |
+
}
|
| 225 |
+
|
| 226 |
+
/**
|
| 227 |
+
* Radio field
|
| 228 |
+
* @param string $key
|
| 229 |
+
* @param mixed $value
|
| 230 |
+
* @param array $attrs Optional
|
| 231 |
+
* @return string
|
| 232 |
+
*/
|
| 233 |
+
public function radio( $key, $value, $attrs = array() ) {
|
| 234 |
+
$checked = ( $value == $this->value( $key ) ) ? ' checked="checked"' : '';
|
| 235 |
+
return '<input type="radio" '. $this->attrs( $attrs, $key, $value )
|
| 236 |
+
. $checked . ' />';
|
| 237 |
+
}
|
| 238 |
+
|
| 239 |
+
/**
|
| 240 |
+
* Checkbox field
|
| 241 |
+
* @param string $key
|
| 242 |
+
* @param mixed $value
|
| 243 |
+
* @param array $attrs Optional
|
| 244 |
+
* @return string
|
| 245 |
+
*/
|
| 246 |
+
public function checkbox( $key, $value, $attrs = array() ) {
|
| 247 |
+
$checked = ( $value == $this->value( $key ) ) ? ' checked="checked"' : '';
|
| 248 |
+
return '<input type="checkbox" '. $this->attrs( $attrs, $key, $value )
|
| 249 |
+
. $checked . ' />';
|
| 250 |
+
}
|
| 251 |
+
|
| 252 |
+
/**
|
| 253 |
+
* Select field
|
| 254 |
+
* @param string $key
|
| 255 |
+
* @param array $options Optional
|
| 256 |
+
* @param array $attrs Optional
|
| 257 |
+
* @return string
|
| 258 |
+
*/
|
| 259 |
+
public function select( $key, $options = array(), $attrs = array() ) {
|
| 260 |
+
$html = '<select '. $this->attrs( $attrs, $key ) .'>';
|
| 261 |
+
|
| 262 |
+
foreach ( $options AS $value => $label ) {
|
| 263 |
+
$selected = ( $value == $this->value( $key ) ) ? ' selected="selected"' : '';
|
| 264 |
+
$html .= '<option value="'. $value .'"'. $selected .'>'. $label .'</option>';
|
| 265 |
+
}
|
| 266 |
+
|
| 267 |
+
$html .= '</select>';
|
| 268 |
+
return $html;
|
| 269 |
+
}
|
| 270 |
+
|
| 271 |
+
/**
|
| 272 |
+
* Submit button
|
| 273 |
+
* @param array $attrs Optional
|
| 274 |
+
* @return string
|
| 275 |
+
*/
|
| 276 |
+
public function submit( $attrs = array() ) {
|
| 277 |
+
// set class attr
|
| 278 |
+
$attrs[ 'class' ] = 'button-primary'. ( ( key_exists( 'class', $attrs ) ) ? ' '. $attrs[ 'class' ] : '' );
|
| 279 |
+
|
| 280 |
+
// show submit
|
| 281 |
+
$html = '';
|
| 282 |
+
$html .= '<p class="button-controls" style="text-align:right;">';
|
| 283 |
+
$html .= '<img alt="" title="" class="ajax-feedback" src="'. get_bloginfo( 'url' ) .'/wp-admin/images/wpspin_light.gif" style="visibility: hidden;" />';
|
| 284 |
+
$html .= '<input type="submit" '. $this->attrs( $attrs, '', __( 'Save Changes' ) ) .' />';
|
| 285 |
+
$html .= '</p>';
|
| 286 |
+
return $html;
|
| 287 |
+
}
|
| 288 |
+
|
| 289 |
+
/**
|
| 290 |
+
* Get field name of given key
|
| 291 |
+
* @param string $key
|
| 292 |
+
* @return string
|
| 293 |
+
*/
|
| 294 |
+
public function field_name( $key ) {
|
| 295 |
+
return $this->current_option . '[' . $key . ']';
|
| 296 |
+
}
|
| 297 |
+
|
| 298 |
+
/**
|
| 299 |
+
* Get value of given option key
|
| 300 |
+
* @param string $key
|
| 301 |
+
* @param mixed $default_value Optional
|
| 302 |
+
* @param boolean $option_name Optional, search in given option_name instead of the current option
|
| 303 |
+
* @return mixed
|
| 304 |
+
*/
|
| 305 |
+
public function value( $key, $default_value = NULL, $option_name = NULL ) {
|
| 306 |
+
if ( $option_name === NULL ) {
|
| 307 |
+
$option = $this->current_option;
|
| 308 |
+
} else {
|
| 309 |
+
$option = $this->name . '-' . $option_name;
|
| 310 |
+
}
|
| 311 |
+
|
| 312 |
+
if (!isset($this->options[ $option ])) {
|
| 313 |
+
return $default_value;
|
| 314 |
+
}
|
| 315 |
+
|
| 316 |
+
$values = $this->options[ $option ];
|
| 317 |
+
|
| 318 |
+
return ( is_array( $values ) AND key_exists( $key, $values ) AND $values[ $key ] !== NULL ) ? $values[ $key ] : $default_value;
|
| 319 |
+
}
|
| 320 |
+
|
| 321 |
+
/**
|
| 322 |
+
* Delete and unregister option
|
| 323 |
+
*/
|
| 324 |
+
public function delete_options() {
|
| 325 |
+
foreach ( $this->options AS $option_name => $values ) {
|
| 326 |
+
delete_option( $option_name );
|
| 327 |
+
}
|
| 328 |
+
}
|
| 329 |
+
|
| 330 |
+
/**
|
| 331 |
+
* Get string of given attributes
|
| 332 |
+
* @param array $attrs
|
| 333 |
+
* @param string $key Optional
|
| 334 |
+
* @param mixed $value Optional
|
| 335 |
+
* @return string
|
| 336 |
+
*/
|
| 337 |
+
protected function attrs( $attrs, $key = NULL, $value = NULL ) {
|
| 338 |
+
$str = '';
|
| 339 |
+
|
| 340 |
+
// set name, id, value attr
|
| 341 |
+
if ( $key !== NULL ) {
|
| 342 |
+
$str .= 'name="' . $this->field_name( $key ) .'" ';
|
| 343 |
+
if ( ! key_exists( 'id', $attrs ) )
|
| 344 |
+
$str .= 'id="' . $key .'" ';
|
| 345 |
+
}
|
| 346 |
+
|
| 347 |
+
if ( $value !== NULL )
|
| 348 |
+
$str .= 'value="' . $value .'" ';
|
| 349 |
+
|
| 350 |
+
foreach ( $attrs AS $attr => $value )
|
| 351 |
+
$str .= $attr .'="'. $value .'" ';
|
| 352 |
+
|
| 353 |
+
return $str;
|
| 354 |
+
}
|
| 355 |
+
|
| 356 |
+
} // End WP_Option_Forms_01
|
| 357 |
+
|
| 358 |
+
endif;
|
| 359 |
+
|
| 360 |
/* ommit PHP closing tag, to prevent unwanted whitespace at the end of the parts generated by the included files */
|
js/admin-wp-external-links.js
CHANGED
|
@@ -1,172 +1,2 @@
|
|
| 1 |
-
/* WP External Links
|
| 2 |
-
jQuery(function (
|
| 3 |
-
|
| 4 |
-
/* Tipsy Plugin */
|
| 5 |
-
(function () {
|
| 6 |
-
$.fn.tipsy = function(options) {
|
| 7 |
-
options = $.extend({}, $.fn.tipsy.defaults, options);
|
| 8 |
-
|
| 9 |
-
return this.each(function() {
|
| 10 |
-
var opts = $.fn.tipsy.elementOptions(this, options);
|
| 11 |
-
|
| 12 |
-
$(this).hover(function() {
|
| 13 |
-
$.data(this, 'cancel.tipsy', true);
|
| 14 |
-
|
| 15 |
-
var tip = $.data(this, 'active.tipsy');
|
| 16 |
-
if (!tip) {
|
| 17 |
-
tip = $('<div class="tipsy"><div class="tipsy-inner"/></div>');
|
| 18 |
-
tip.css({position: 'absolute', zIndex: 100000});
|
| 19 |
-
$.data(this, 'active.tipsy', tip);
|
| 20 |
-
}
|
| 21 |
-
|
| 22 |
-
if ($(this).attr('title') || typeof($(this).attr('original-title')) != 'string') {
|
| 23 |
-
$(this).attr('original-title', $(this).attr('title') || '').removeAttr('title');
|
| 24 |
-
}
|
| 25 |
-
|
| 26 |
-
var title;
|
| 27 |
-
if (typeof opts.title == 'string') {
|
| 28 |
-
title = $(this).attr(opts.title == 'title' ? 'original-title' : opts.title);
|
| 29 |
-
} else if (typeof opts.title == 'function') {
|
| 30 |
-
title = opts.title.call(this);
|
| 31 |
-
}
|
| 32 |
-
|
| 33 |
-
tip.find('.tipsy-inner')[opts.html ? 'html' : 'text'](title || opts.fallback);
|
| 34 |
-
|
| 35 |
-
var pos = $.extend({}, $(this).offset(), {width: this.offsetWidth, height: this.offsetHeight});
|
| 36 |
-
tip.get(0).className = 'tipsy'; // reset classname in case of dynamic gravity
|
| 37 |
-
tip.remove().css({top: 0, left: 0, visibility: 'hidden', display: 'block'}).appendTo(document.body);
|
| 38 |
-
var actualWidth = tip[0].offsetWidth, actualHeight = tip[0].offsetHeight;
|
| 39 |
-
var gravity = (typeof opts.gravity == 'function') ? opts.gravity.call(this) : opts.gravity;
|
| 40 |
-
|
| 41 |
-
switch (gravity.charAt(0)) {
|
| 42 |
-
case 'n':
|
| 43 |
-
tip.css({top: pos.top + pos.height, left: pos.left + pos.width / 2 - actualWidth / 2}).addClass('tipsy-north');
|
| 44 |
-
break;
|
| 45 |
-
case 's':
|
| 46 |
-
tip.css({top: pos.top - actualHeight, left: pos.left + pos.width / 2 - actualWidth / 2}).addClass('tipsy-south');
|
| 47 |
-
break;
|
| 48 |
-
case 'e':
|
| 49 |
-
tip.css({top: pos.top + pos.height / 2 - actualHeight / 2, left: pos.left - actualWidth}).addClass('tipsy-east');
|
| 50 |
-
break;
|
| 51 |
-
case 'w':
|
| 52 |
-
tip.css({top: pos.top + pos.height / 2 - actualHeight / 2, left: pos.left + pos.width}).addClass('tipsy-west');
|
| 53 |
-
break;
|
| 54 |
-
}
|
| 55 |
-
|
| 56 |
-
if (opts.fade) {
|
| 57 |
-
tip.css({opacity: 0, display: 'block', visibility: 'visible'}).animate({opacity: 0.9});
|
| 58 |
-
} else {
|
| 59 |
-
tip.css({visibility: 'visible'});
|
| 60 |
-
}
|
| 61 |
-
|
| 62 |
-
}, function() {
|
| 63 |
-
$.data(this, 'cancel.tipsy', false);
|
| 64 |
-
var self = this;
|
| 65 |
-
setTimeout(function() {
|
| 66 |
-
if ($.data(this, 'cancel.tipsy')) return;
|
| 67 |
-
var tip = $.data(self, 'active.tipsy');
|
| 68 |
-
if (opts.fade) {
|
| 69 |
-
tip.stop().fadeOut(function() { $(this).remove(); });
|
| 70 |
-
} else {
|
| 71 |
-
tip.remove();
|
| 72 |
-
}
|
| 73 |
-
}, 100);
|
| 74 |
-
|
| 75 |
-
});
|
| 76 |
-
});
|
| 77 |
-
};
|
| 78 |
-
|
| 79 |
-
// Overwrite this method to provide options on a per-element basis.
|
| 80 |
-
// For example, you could store the gravity in a 'tipsy-gravity' attribute:
|
| 81 |
-
// return $.extend({}, options, {gravity: $(ele).attr('tipsy-gravity') || 'n' });
|
| 82 |
-
// (remember - do not modify 'options' in place!)
|
| 83 |
-
$.fn.tipsy.elementOptions = function(ele, options) {
|
| 84 |
-
return $.metadata ? $.extend({}, options, $(ele).metadata()) : options;
|
| 85 |
-
};
|
| 86 |
-
|
| 87 |
-
$.fn.tipsy.defaults = {
|
| 88 |
-
fade: false,
|
| 89 |
-
fallback: '',
|
| 90 |
-
gravity: 'w',
|
| 91 |
-
html: false,
|
| 92 |
-
title: 'title'
|
| 93 |
-
};
|
| 94 |
-
|
| 95 |
-
$.fn.tipsy.autoNS = function() {
|
| 96 |
-
return $(this).offset().top > ($(document).scrollTop() + $(window).height() / 2) ? 's' : 'n';
|
| 97 |
-
};
|
| 98 |
-
|
| 99 |
-
$.fn.tipsy.autoWE = function() {
|
| 100 |
-
return $(this).offset().left > ($(document).scrollLeft() + $(window).width() / 2) ? 'e' : 'w';
|
| 101 |
-
};
|
| 102 |
-
|
| 103 |
-
})(); // End Tipsy Plugin
|
| 104 |
-
|
| 105 |
-
|
| 106 |
-
$('#setting-error-settings_updated').click(function () {
|
| 107 |
-
$(this).hide();
|
| 108 |
-
});
|
| 109 |
-
|
| 110 |
-
// option filter page
|
| 111 |
-
$( 'input#filter_page' )
|
| 112 |
-
.change(function(){
|
| 113 |
-
var $i = $( 'input#filter_posts, input#filter_comments, input#filter_widgets' );
|
| 114 |
-
|
| 115 |
-
if ( $( this ).attr( 'checked' ) ) {
|
| 116 |
-
$i.attr( 'disabled', true )
|
| 117 |
-
.attr( 'checked', true );
|
| 118 |
-
} else {
|
| 119 |
-
$i.attr( 'disabled', false )
|
| 120 |
-
}
|
| 121 |
-
})
|
| 122 |
-
.change();
|
| 123 |
-
|
| 124 |
-
// option use js
|
| 125 |
-
$( 'input#use_js' )
|
| 126 |
-
.change(function(){
|
| 127 |
-
var $i = $( 'input#load_in_footer' );
|
| 128 |
-
|
| 129 |
-
if ( $( this ).attr( 'checked' ) ) {
|
| 130 |
-
$i.attr( 'disabled', false );
|
| 131 |
-
} else {
|
| 132 |
-
$i.attr( 'disabled', true )
|
| 133 |
-
.attr( 'checked', false );
|
| 134 |
-
}
|
| 135 |
-
})
|
| 136 |
-
.change();
|
| 137 |
-
|
| 138 |
-
// option filter_excl_sel
|
| 139 |
-
$( 'input#phpquery' )
|
| 140 |
-
.change(function(){
|
| 141 |
-
if ( $( this ).attr( 'checked' ) ) {
|
| 142 |
-
$( '.filter_excl_sel' ).fadeIn();
|
| 143 |
-
} else {
|
| 144 |
-
$( '.filter_excl_sel' ).fadeOut();
|
| 145 |
-
}
|
| 146 |
-
})
|
| 147 |
-
.change();
|
| 148 |
-
|
| 149 |
-
// refresh page when updated menu position
|
| 150 |
-
$('#menu_position').parents( 'form.ajax-form' ).on( 'ajax_saved_options', function(result){
|
| 151 |
-
var s = $( this ).val() || '';
|
| 152 |
-
window.location.href = s + ( s.indexOf( '?' ) > -1 ? '&' : '?' ) + 'page=wp_external_links&settings-updated=true';
|
| 153 |
-
});
|
| 154 |
-
|
| 155 |
-
// set tooltips
|
| 156 |
-
$( '.tooltip-help' ).css( 'margin', '0 5px' ).tipsy({ fade:true, live:true, gravity:'w', fallback: 'No help text.' });
|
| 157 |
-
|
| 158 |
-
// remove class to fix button background
|
| 159 |
-
$('*[type="submit"]').removeClass('submit');
|
| 160 |
-
|
| 161 |
-
// slide postbox
|
| 162 |
-
$( '.postbox' ).find( '.handlediv, .hndle' ).click(function(){
|
| 163 |
-
var $inside = $( this ).parent().find( '.inside' );
|
| 164 |
-
|
| 165 |
-
if ( $inside.css( 'display' ) == 'block' ) {
|
| 166 |
-
$inside.css({ display:'none' });
|
| 167 |
-
} else {
|
| 168 |
-
$inside.css({ display:'block' });
|
| 169 |
-
}
|
| 170 |
-
});
|
| 171 |
-
|
| 172 |
-
});
|
| 1 |
+
/* WP External Links - Admin */
|
| 2 |
+
jQuery(function(e){"use strict";(function(){e.fn.tipsy=function(t){t=e.extend({},e.fn.tipsy.defaults,t);return this.each(function(){var n=e.fn.tipsy.elementOptions(this,t);e(this).hover(function(){e.data(this,"cancel.tipsy",true);var t=e.data(this,"active.tipsy");if(!t){t=e('<div class="tipsy"><div class="tipsy-inner"/></div>');t.css({position:"absolute",zIndex:1e5});e.data(this,"active.tipsy",t)}if(e(this).attr("title")||typeof e(this).attr("original-title")!=="string"){e(this).attr("original-title",e(this).attr("title")||"").removeAttr("title")}var r;if(typeof n.title==="string"){r=e(this).attr(n.title==="title"?"original-title":n.title)}else if(typeof n.title==="function"){r=n.title.call(this)}t.find(".tipsy-inner")[n.html?"html":"text"](r||n.fallback);var i=e.extend({},e(this).offset(),{width:this.offsetWidth,height:this.offsetHeight});t.get(0).className="tipsy";t.remove().css({top:0,left:0,visibility:"hidden",display:"block"}).appendTo(document.body);var s=t[0].offsetWidth,o=t[0].offsetHeight;var u=typeof n.gravity=="function"?n.gravity.call(this):n.gravity;switch(u.charAt(0)){case"n":t.css({top:i.top+i.height,left:i.left+i.width/2-s/2}).addClass("tipsy-north");break;case"s":t.css({top:i.top-o,left:i.left+i.width/2-s/2}).addClass("tipsy-south");break;case"e":t.css({top:i.top+i.height/2-o/2,left:i.left-s}).addClass("tipsy-east");break;case"w":t.css({top:i.top+i.height/2-o/2,left:i.left+i.width}).addClass("tipsy-west");break}if(n.fade){t.css({opacity:0,display:"block",visibility:"visible"}).animate({opacity:.9})}else{t.css({visibility:"visible"})}},function(){e.data(this,"cancel.tipsy",false);var t=this;setTimeout(function(){if(e.data(this,"cancel.tipsy"))return;var r=e.data(t,"active.tipsy");if(n.fade){r.stop().fadeOut(function(){e(this).remove()})}else{r.remove()}},100)})})};e.fn.tipsy.elementOptions=function(t,n){return e.metadata?e.extend({},n,e(t).metadata()):n};e.fn.tipsy.defaults={fade:false,fallback:"",gravity:"w",html:false,title:"title"};e.fn.tipsy.autoNS=function(){return e(this).offset().top>e(document).scrollTop()+e(window).height()/2?"s":"n"};e.fn.tipsy.autoWE=function(){return e(this).offset().left>e(document).scrollLeft()+e(window).width()/2?"e":"w"}})();e("#setting-error-settings_updated").click(function(){e(this).hide()});e("input#filter_page").change(function(){var t=e("input#filter_posts, input#filter_comments, input#filter_widgets");if(e(this).attr("checked")){t.attr("disabled",true).attr("checked",true)}else{t.attr("disabled",false)}}).change();e("input#use_js").change(function(){var t=e("input#load_in_footer");if(e(this).attr("checked")){t.attr("disabled",false)}else{t.attr("disabled",true).attr("checked",false)}}).change();e("input#phpquery").change(function(){if(e(this).attr("checked")){e(".filter_excl_sel").fadeIn()}else{e(".filter_excl_sel").fadeOut()}}).change();e("#menu_position").parents("form.ajax-form").on("ajax_saved_options",function(){var t=e(this).val()||"";window.location.href=t+(t.indexOf("?")>-1?"&":"?")+"page=wp_external_links&settings-updated=true"});e(".tooltip-help").css("margin","0 5px").tipsy({fade:true,live:true,gravity:"w",fallback:"No help text."});e('*[type="submit"]').removeClass("submit");e(".postbox").find(".handlediv, .hndle").click(function(){var t=e(this).parent().find(".inside");if(t.css("display")==="block"){t.css({display:"none"})}else{t.css({display:"block"})}})});jQuery(function(e){"use strict";var t=function(t){var n=e(t),r=n.parents("form"),i=r.serializeArray();n.attr("disabled",true);r.find(".ajax-feedback").css("visibility","visible");e.post(ajaxurl,i,function(t){var i=e("<strong>").insertBefore(n);if(t==="1"){i.html("Saved")}else{r.find('[name="action"]').val("update");r.submit()}i.css({margin:"0 5px"}).delay(1e3).fadeOut(function(){e(this).remove()});n.attr("disabled",false);r.find(".ajax-feedback").css("visibility","hidden");r.trigger("ajax_saved_options",[t])})};e('form.ajax-form input[type="submit"]').click(function(e){t(this);e.preventDefault()})})
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
js/src/admin-wp-external-links.js
ADDED
|
@@ -0,0 +1,173 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
/* WP External Links Plugin - Admin */
|
| 2 |
+
/*global jQuery, window*/
|
| 3 |
+
jQuery(function ($) {
|
| 4 |
+
'use strict';
|
| 5 |
+
|
| 6 |
+
/* Tipsy Plugin */
|
| 7 |
+
(function () {
|
| 8 |
+
$.fn.tipsy = function (options) {
|
| 9 |
+
options = $.extend({}, $.fn.tipsy.defaults, options);
|
| 10 |
+
|
| 11 |
+
return this.each(function () {
|
| 12 |
+
var opts = $.fn.tipsy.elementOptions(this, options);
|
| 13 |
+
|
| 14 |
+
$(this).hover(function () {
|
| 15 |
+
$.data(this, 'cancel.tipsy', true);
|
| 16 |
+
|
| 17 |
+
var tip = $.data(this, 'active.tipsy');
|
| 18 |
+
if (!tip) {
|
| 19 |
+
tip = $('<div class="tipsy"><div class="tipsy-inner"/></div>');
|
| 20 |
+
tip.css({position: 'absolute', zIndex: 100000});
|
| 21 |
+
$.data(this, 'active.tipsy', tip);
|
| 22 |
+
}
|
| 23 |
+
|
| 24 |
+
if ($(this).attr('title') || typeof $(this).attr('original-title') !== 'string') {
|
| 25 |
+
$(this).attr('original-title', $(this).attr('title') || '').removeAttr('title');
|
| 26 |
+
}
|
| 27 |
+
|
| 28 |
+
var title;
|
| 29 |
+
if (typeof opts.title === 'string') {
|
| 30 |
+
title = $(this).attr(opts.title === 'title' ? 'original-title' : opts.title);
|
| 31 |
+
} else if (typeof opts.title === 'function') {
|
| 32 |
+
title = opts.title.call(this);
|
| 33 |
+
}
|
| 34 |
+
|
| 35 |
+
tip.find('.tipsy-inner')[opts.html ? 'html' : 'text'](title || opts.fallback);
|
| 36 |
+
|
| 37 |
+
var pos = $.extend({}, $(this).offset(), {width: this.offsetWidth, height: this.offsetHeight});
|
| 38 |
+
tip.get(0).className = 'tipsy'; // reset classname in case of dynamic gravity
|
| 39 |
+
tip.remove().css({top: 0, left: 0, visibility: 'hidden', display: 'block'}).appendTo(document.body);
|
| 40 |
+
var actualWidth = tip[0].offsetWidth, actualHeight = tip[0].offsetHeight;
|
| 41 |
+
var gravity = (typeof opts.gravity == 'function') ? opts.gravity.call(this) : opts.gravity;
|
| 42 |
+
|
| 43 |
+
switch (gravity.charAt(0)) {
|
| 44 |
+
case 'n':
|
| 45 |
+
tip.css({top: pos.top + pos.height, left: pos.left + pos.width / 2 - actualWidth / 2}).addClass('tipsy-north');
|
| 46 |
+
break;
|
| 47 |
+
case 's':
|
| 48 |
+
tip.css({top: pos.top - actualHeight, left: pos.left + pos.width / 2 - actualWidth / 2}).addClass('tipsy-south');
|
| 49 |
+
break;
|
| 50 |
+
case 'e':
|
| 51 |
+
tip.css({top: pos.top + pos.height / 2 - actualHeight / 2, left: pos.left - actualWidth}).addClass('tipsy-east');
|
| 52 |
+
break;
|
| 53 |
+
case 'w':
|
| 54 |
+
tip.css({top: pos.top + pos.height / 2 - actualHeight / 2, left: pos.left + pos.width}).addClass('tipsy-west');
|
| 55 |
+
break;
|
| 56 |
+
}
|
| 57 |
+
|
| 58 |
+
if (opts.fade) {
|
| 59 |
+
tip.css({opacity: 0, display: 'block', visibility: 'visible'}).animate({opacity: 0.9});
|
| 60 |
+
} else {
|
| 61 |
+
tip.css({visibility: 'visible'});
|
| 62 |
+
}
|
| 63 |
+
|
| 64 |
+
}, function () {
|
| 65 |
+
$.data(this, 'cancel.tipsy', false);
|
| 66 |
+
var self = this;
|
| 67 |
+
setTimeout(function () {
|
| 68 |
+
if ($.data(this, 'cancel.tipsy')) return;
|
| 69 |
+
var tip = $.data(self, 'active.tipsy');
|
| 70 |
+
if (opts.fade) {
|
| 71 |
+
tip.stop().fadeOut(function () { $(this).remove(); });
|
| 72 |
+
} else {
|
| 73 |
+
tip.remove();
|
| 74 |
+
}
|
| 75 |
+
}, 100);
|
| 76 |
+
});
|
| 77 |
+
});
|
| 78 |
+
};
|
| 79 |
+
|
| 80 |
+
// Overwrite this method to provide options on a per-element basis.
|
| 81 |
+
// For example, you could store the gravity in a 'tipsy-gravity' attribute:
|
| 82 |
+
// return $.extend({}, options, {gravity: $(ele).attr('tipsy-gravity') || 'n' });
|
| 83 |
+
// (remember - do not modify 'options' in place!)
|
| 84 |
+
$.fn.tipsy.elementOptions = function (ele, options) {
|
| 85 |
+
return $.metadata ? $.extend({}, options, $(ele).metadata()) : options;
|
| 86 |
+
};
|
| 87 |
+
|
| 88 |
+
$.fn.tipsy.defaults = {
|
| 89 |
+
fade: false,
|
| 90 |
+
fallback: '',
|
| 91 |
+
gravity: 'w',
|
| 92 |
+
html: false,
|
| 93 |
+
title: 'title'
|
| 94 |
+
};
|
| 95 |
+
|
| 96 |
+
$.fn.tipsy.autoNS = function () {
|
| 97 |
+
return $(this).offset().top > ($(document).scrollTop() + $(window).height() / 2) ? 's' : 'n';
|
| 98 |
+
};
|
| 99 |
+
|
| 100 |
+
$.fn.tipsy.autoWE = function () {
|
| 101 |
+
return $(this).offset().left > ($(document).scrollLeft() + $(window).width() / 2) ? 'e' : 'w';
|
| 102 |
+
};
|
| 103 |
+
|
| 104 |
+
})(); // End Tipsy Plugin
|
| 105 |
+
|
| 106 |
+
|
| 107 |
+
$('#setting-error-settings_updated').click(function () {
|
| 108 |
+
$(this).hide();
|
| 109 |
+
});
|
| 110 |
+
|
| 111 |
+
// option filter page
|
| 112 |
+
$('input#filter_page')
|
| 113 |
+
.change(function () {
|
| 114 |
+
var $i = $('input#filter_posts, input#filter_comments, input#filter_widgets');
|
| 115 |
+
|
| 116 |
+
if ($(this).attr('checked')) {
|
| 117 |
+
$i.attr('disabled', true)
|
| 118 |
+
.attr('checked', true);
|
| 119 |
+
} else {
|
| 120 |
+
$i.attr('disabled', false);
|
| 121 |
+
}
|
| 122 |
+
})
|
| 123 |
+
.change();
|
| 124 |
+
|
| 125 |
+
// option use js
|
| 126 |
+
$('input#use_js')
|
| 127 |
+
.change(function () {
|
| 128 |
+
var $i = $('input#load_in_footer');
|
| 129 |
+
|
| 130 |
+
if ($(this).attr('checked')) {
|
| 131 |
+
$i.attr('disabled', false);
|
| 132 |
+
} else {
|
| 133 |
+
$i.attr('disabled', true)
|
| 134 |
+
.attr('checked', false);
|
| 135 |
+
}
|
| 136 |
+
})
|
| 137 |
+
.change();
|
| 138 |
+
|
| 139 |
+
// option filter_excl_sel
|
| 140 |
+
$('input#phpquery')
|
| 141 |
+
.change(function () {
|
| 142 |
+
if ($(this).attr('checked')) {
|
| 143 |
+
$('.filter_excl_sel').fadeIn();
|
| 144 |
+
} else {
|
| 145 |
+
$('.filter_excl_sel').fadeOut();
|
| 146 |
+
}
|
| 147 |
+
})
|
| 148 |
+
.change();
|
| 149 |
+
|
| 150 |
+
// refresh page when updated menu position
|
| 151 |
+
$('#menu_position').parents('form.ajax-form').on('ajax_saved_options', function () {
|
| 152 |
+
var s = $(this).val() || '';
|
| 153 |
+
window.location.href = s + (s.indexOf('?') > -1 ? '&' : '?') + 'page=wp_external_links&settings-updated=true';
|
| 154 |
+
});
|
| 155 |
+
|
| 156 |
+
// set tooltips
|
| 157 |
+
$('.tooltip-help').css('margin', '0 5px').tipsy({ fade: true, live: true, gravity: 'w', fallback: 'No help text.' });
|
| 158 |
+
|
| 159 |
+
// remove class to fix button background
|
| 160 |
+
$('*[type="submit"]').removeClass('submit');
|
| 161 |
+
|
| 162 |
+
// slide postbox
|
| 163 |
+
$('.postbox').find('.handlediv, .hndle').click(function () {
|
| 164 |
+
var $inside = $(this).parent().find('.inside');
|
| 165 |
+
|
| 166 |
+
if ($inside.css('display') === 'block') {
|
| 167 |
+
$inside.css({ display: 'none' });
|
| 168 |
+
} else {
|
| 169 |
+
$inside.css({ display: 'block' });
|
| 170 |
+
}
|
| 171 |
+
});
|
| 172 |
+
|
| 173 |
+
});
|
js/src/wp-external-links.js
ADDED
|
@@ -0,0 +1,75 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
/* WP External Links Plugin */
|
| 2 |
+
/*global jQuery, window, wpExtLinks*/
|
| 3 |
+
(function () {
|
| 4 |
+
'use strict';
|
| 5 |
+
|
| 6 |
+
var $ = jQuery === undefined ? null : jQuery;
|
| 7 |
+
|
| 8 |
+
// add event handler
|
| 9 |
+
function addEvt(el, evt, fn) {
|
| 10 |
+
if (el.attachEvent) {
|
| 11 |
+
// IE method
|
| 12 |
+
el.attachEvent('on' + evt, fn);
|
| 13 |
+
} else if (el.addEventListener) {
|
| 14 |
+
// Standard JS method
|
| 15 |
+
el.addEventListener(evt, fn, false);
|
| 16 |
+
}
|
| 17 |
+
}
|
| 18 |
+
|
| 19 |
+
// open external link
|
| 20 |
+
function openExtLink(a, opts, e) {
|
| 21 |
+
var options = opts || wpExtLinks,
|
| 22 |
+
href = a.href ? a.href.toLowerCase() : '',
|
| 23 |
+
rel = a.rel ? a.rel.toLowerCase() : '',
|
| 24 |
+
n;
|
| 25 |
+
|
| 26 |
+
if (a.href && (options.excludeClass.length === 0 || a.className.indexOf(options.excludeClass))
|
| 27 |
+
&& (rel.indexOf('external') > -1
|
| 28 |
+
|| ((href.indexOf(options.baseUrl) === -1) &&
|
| 29 |
+
(href.substr(0, 7) === 'http://'
|
| 30 |
+
|| href.substr(0, 8) === 'https://'
|
| 31 |
+
|| href.substr(0, 6) === 'ftp://'
|
| 32 |
+
|| href.substr(0, 2) === '//')))) {
|
| 33 |
+
// open link in a new window
|
| 34 |
+
n = window.open(a.href, options.target);
|
| 35 |
+
n.focus();
|
| 36 |
+
|
| 37 |
+
// prevent default event action
|
| 38 |
+
if (e) {
|
| 39 |
+
if (e.preventDefault) {
|
| 40 |
+
e.preventDefault();
|
| 41 |
+
} else {
|
| 42 |
+
e.returnValue = false;
|
| 43 |
+
}
|
| 44 |
+
}
|
| 45 |
+
}
|
| 46 |
+
}
|
| 47 |
+
|
| 48 |
+
if ($ && false) {
|
| 49 |
+
// jQuery DOMready method
|
| 50 |
+
$(function () {
|
| 51 |
+
$('a').live('click', function (e) {
|
| 52 |
+
openExtLink(this, null, e);
|
| 53 |
+
});
|
| 54 |
+
});
|
| 55 |
+
} else {
|
| 56 |
+
// use onload when jQuery not available
|
| 57 |
+
addEvt(window, 'load', function () {
|
| 58 |
+
var links = window.document.getElementsByTagName('a'),
|
| 59 |
+
eventClick = function (e) {
|
| 60 |
+
openExtLink(e.target, null, e);
|
| 61 |
+
},
|
| 62 |
+
a,
|
| 63 |
+
i;
|
| 64 |
+
|
| 65 |
+
// check each <a> element
|
| 66 |
+
for (i = 0; i < links.length; i += 1) {
|
| 67 |
+
a = links[i];
|
| 68 |
+
|
| 69 |
+
// click event for opening in a new window
|
| 70 |
+
addEvt(a, 'click', eventClick);
|
| 71 |
+
}
|
| 72 |
+
});
|
| 73 |
+
}
|
| 74 |
+
|
| 75 |
+
}());
|
js/src/wp-option-forms.js
ADDED
|
@@ -0,0 +1,55 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
/* WP Options Form */
|
| 2 |
+
/*global jQuery, ajaxurl*/
|
| 3 |
+
jQuery(function ($) {
|
| 4 |
+
'use strict';
|
| 5 |
+
|
| 6 |
+
// save function
|
| 7 |
+
var saveAjaxForm = function (target) {
|
| 8 |
+
var $this = $(target),
|
| 9 |
+
$form = $this.parents('form'),
|
| 10 |
+
// get ajax post values
|
| 11 |
+
vals = $form.serializeArray();
|
| 12 |
+
|
| 13 |
+
// disable button
|
| 14 |
+
$this.attr('disabled', true);
|
| 15 |
+
|
| 16 |
+
// show ajax loader
|
| 17 |
+
$form.find('.ajax-feedback').css('visibility', 'visible');
|
| 18 |
+
|
| 19 |
+
// save option values
|
| 20 |
+
$.post(ajaxurl, vals, function (result) {
|
| 21 |
+
var $msg = $('<strong>').insertBefore($this);
|
| 22 |
+
|
| 23 |
+
if (result === '1') {
|
| 24 |
+
$msg.html('Saved');
|
| 25 |
+
} else {
|
| 26 |
+
// save options, non-ajax fallback
|
| 27 |
+
$form.find('[name="action"]').val('update');
|
| 28 |
+
// normal submit
|
| 29 |
+
$form.submit();
|
| 30 |
+
}
|
| 31 |
+
|
| 32 |
+
$msg.css({ margin: '0 5px' })
|
| 33 |
+
.delay(1000)
|
| 34 |
+
.fadeOut(function () {
|
| 35 |
+
$(this).remove();
|
| 36 |
+
});
|
| 37 |
+
|
| 38 |
+
// enable button
|
| 39 |
+
$this.attr('disabled', false);
|
| 40 |
+
|
| 41 |
+
// hide ajax loader
|
| 42 |
+
$form.find('.ajax-feedback').css('visibility', 'hidden');
|
| 43 |
+
|
| 44 |
+
// trigger ajax_saved_options
|
| 45 |
+
$form.trigger('ajax_saved_options', [result]);
|
| 46 |
+
});
|
| 47 |
+
};
|
| 48 |
+
|
| 49 |
+
// add ajax post
|
| 50 |
+
$('form.ajax-form input[type="submit"]').click(function (e) {
|
| 51 |
+
saveAjaxForm(this);
|
| 52 |
+
e.preventDefault();
|
| 53 |
+
});
|
| 54 |
+
|
| 55 |
+
});
|
js/wp-external-links.js
CHANGED
|
@@ -1,72 +1,2 @@
|
|
| 1 |
-
/* WP External Links
|
| 2 |
-
(function(){
|
| 3 |
-
|
| 4 |
-
var $ = typeof jQuery == 'undefined' ? null : jQuery;
|
| 5 |
-
|
| 6 |
-
// add event handler
|
| 7 |
-
function addEvt ( el, evt, fn ) {
|
| 8 |
-
if ( el.attachEvent ) {
|
| 9 |
-
// IE method
|
| 10 |
-
el.attachEvent( 'on'+ evt, fn );
|
| 11 |
-
} else if ( el.addEventListener ) {
|
| 12 |
-
// Standard JS method
|
| 13 |
-
el.addEventListener( evt, fn, false );
|
| 14 |
-
}
|
| 15 |
-
};
|
| 16 |
-
|
| 17 |
-
// open external link
|
| 18 |
-
function openExtLink( a, opts, e ) {
|
| 19 |
-
var options = opts ? opts : wpExtLinks,
|
| 20 |
-
href = a.href ? a.href.toLowerCase() : '',
|
| 21 |
-
rel = a.rel ? a.rel.toLowerCase() : '',
|
| 22 |
-
n;
|
| 23 |
-
|
| 24 |
-
if ( a.href && ( options.excludeClass.length == 0 || a.className.indexOf( options.excludeClass ) )
|
| 25 |
-
&& ( rel.indexOf( 'external' ) > -1
|
| 26 |
-
|| ( ( href.indexOf( options.baseUrl ) === -1 ) &&
|
| 27 |
-
( href.substr( 0, 7 ) == 'http://'
|
| 28 |
-
|| href.substr( 0, 8 ) == 'https://'
|
| 29 |
-
|| href.substr( 0, 6 ) == 'ftp://' ) ) ) ) {
|
| 30 |
-
// open link in a new window
|
| 31 |
-
n = window.open( a.href, options.target );
|
| 32 |
-
n.focus();
|
| 33 |
-
|
| 34 |
-
// prevent default event action
|
| 35 |
-
if ( e ) {
|
| 36 |
-
if ( e.preventDefault ) {
|
| 37 |
-
e.preventDefault();
|
| 38 |
-
} else {
|
| 39 |
-
e.returnValue = false;
|
| 40 |
-
}
|
| 41 |
-
}
|
| 42 |
-
}
|
| 43 |
-
};
|
| 44 |
-
|
| 45 |
-
if ( $ ) {
|
| 46 |
-
// jQuery DOMready method
|
| 47 |
-
$(function(){
|
| 48 |
-
$( 'a' ).live( 'click', function( e ){
|
| 49 |
-
openExtLink( this, null, e );
|
| 50 |
-
});
|
| 51 |
-
});
|
| 52 |
-
} else {
|
| 53 |
-
// use onload when jQuery not available
|
| 54 |
-
addEvt( window, 'load', function () {
|
| 55 |
-
var links = window.document.getElementsByTagName( 'a' ),
|
| 56 |
-
a;
|
| 57 |
-
|
| 58 |
-
// check each <a> element
|
| 59 |
-
for ( var i = 0; i < links.length; i++ ) {
|
| 60 |
-
a = links[ i ];
|
| 61 |
-
|
| 62 |
-
// click event for opening in a new window
|
| 63 |
-
addEvt( a, 'click', function( a ){
|
| 64 |
-
return function ( e ) {
|
| 65 |
-
openExtLink( a, null, e );
|
| 66 |
-
}
|
| 67 |
-
}( a ));
|
| 68 |
-
}
|
| 69 |
-
});
|
| 70 |
-
}
|
| 71 |
-
|
| 72 |
-
})();
|
| 1 |
+
/* WP External Links */
|
| 2 |
+
(function(){"use strict";function t(e,t,n){if(e.attachEvent){e.attachEvent("on"+t,n)}else if(e.addEventListener){e.addEventListener(t,n,false)}}function n(e,t,n){var r=t||wpExtLinks,i=e.href?e.href.toLowerCase():"",s=e.rel?e.rel.toLowerCase():"",o;if(e.href&&(r.excludeClass.length===0||e.className.indexOf(r.excludeClass))&&(s.indexOf("external")>-1||i.indexOf(r.baseUrl)===-1&&(i.substr(0,7)==="http://"||i.substr(0,8)==="https://"||i.substr(0,6)==="ftp://"||i.substr(0,2)==="//"))){o=window.open(e.href,r.target);o.focus();if(n){if(n.preventDefault){n.preventDefault()}else{n.returnValue=false}}}}var e=jQuery===undefined?null:jQuery;if(e){e(function(){e("a").live("click",function(e){n(this,null,e)})})}else{t(window,"load",function(){var e=window.document.getElementsByTagName("a"),r=function(e){n(e.target,null,e)},i,s;for(s=0;s<e.length;s+=1){i=e[s];t(i,"click",r)}})}})()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
js/wp-option-forms.js
DELETED
|
@@ -1,55 +0,0 @@
|
|
| 1 |
-
/* WP Options Form */
|
| 2 |
-
/*global jQuery, ajaxurl*/
|
| 3 |
-
jQuery(function ($) {
|
| 4 |
-
'use strict';
|
| 5 |
-
|
| 6 |
-
// save function
|
| 7 |
-
var saveAjaxForm = function (target) {
|
| 8 |
-
var $this = $(target),
|
| 9 |
-
$form = $this.parents('form'),
|
| 10 |
-
// get ajax post values
|
| 11 |
-
vals = $form.serializeArray();
|
| 12 |
-
|
| 13 |
-
// disable button
|
| 14 |
-
$this.attr('disabled', true);
|
| 15 |
-
|
| 16 |
-
// show ajax loader
|
| 17 |
-
$form.find('.ajax-feedback').css('visibility', 'visible');
|
| 18 |
-
|
| 19 |
-
// save option values
|
| 20 |
-
$.post(ajaxurl, vals, function (result) {
|
| 21 |
-
var $msg = $('<strong>').insertBefore($this);
|
| 22 |
-
|
| 23 |
-
if (result === '1') {
|
| 24 |
-
$msg.html('Saved');
|
| 25 |
-
} else {
|
| 26 |
-
// save options, non-ajax fallback
|
| 27 |
-
$form.find('[name="action"]').val('update');
|
| 28 |
-
// normal submit
|
| 29 |
-
$form.submit();
|
| 30 |
-
}
|
| 31 |
-
|
| 32 |
-
$msg.css({ margin: '0 5px' })
|
| 33 |
-
.delay(1000)
|
| 34 |
-
.fadeOut(function(){
|
| 35 |
-
$(this).remove();
|
| 36 |
-
});
|
| 37 |
-
|
| 38 |
-
// enable button
|
| 39 |
-
$this.attr('disabled', false);
|
| 40 |
-
|
| 41 |
-
// hide ajax loader
|
| 42 |
-
$form.find('.ajax-feedback').css('visibility', 'hidden');
|
| 43 |
-
|
| 44 |
-
// trigger ajax_saved_options
|
| 45 |
-
$form.trigger('ajax_saved_options', [result]);
|
| 46 |
-
});
|
| 47 |
-
};
|
| 48 |
-
|
| 49 |
-
// add ajax post
|
| 50 |
-
$('form.ajax-form input[type="submit"]').click(function(e) {
|
| 51 |
-
saveAjaxForm(this);
|
| 52 |
-
e.preventDefault();
|
| 53 |
-
});
|
| 54 |
-
|
| 55 |
-
});
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
readme.txt
CHANGED
|
@@ -1,200 +1,212 @@
|
|
| 1 |
-
=== WP External Links (nofollow new window seo) ===
|
| 2 |
-
Contributors: freelancephp
|
| 3 |
-
Tags: links, external, icon, target, _blank, _new, _none, rel, nofollow, new window, new tab, javascript, xhtml, seo
|
| 4 |
-
Requires at least: 3.4.0
|
| 5 |
-
Tested up to: 3.8.2
|
| 6 |
-
Stable tag: 1.
|
| 7 |
-
|
| 8 |
-
Open external links in a new window or tab, adding "nofollow", set link icon, styling, SEO friendly options and more. Easy install and go.
|
| 9 |
-
|
| 10 |
-
== Description ==
|
| 11 |
-
|
| 12 |
-
Configure settings for all external links on your site.
|
| 13 |
-
|
| 14 |
-
= Features =
|
| 15 |
-
* Open
|
| 16 |
-
* Add "nofollow"
|
| 17 |
-
*
|
| 18 |
-
* Set link
|
| 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 |
-
* Fixed
|
| 96 |
-
* Fixed
|
| 97 |
-
|
| 98 |
-
|
| 99 |
-
|
| 100 |
-
|
| 101 |
-
|
| 102 |
-
|
| 103 |
-
|
| 104 |
-
*
|
| 105 |
-
*
|
| 106 |
-
* Fixed
|
| 107 |
-
*
|
| 108 |
-
|
| 109 |
-
|
| 110 |
-
|
| 111 |
-
* Fixed
|
| 112 |
-
|
| 113 |
-
= 1.
|
| 114 |
-
*
|
| 115 |
-
* Added
|
| 116 |
-
|
| 117 |
-
|
| 118 |
-
* Fixed
|
| 119 |
-
*
|
| 120 |
-
|
| 121 |
-
= 1.
|
| 122 |
-
*
|
| 123 |
-
*
|
| 124 |
-
|
| 125 |
-
|
| 126 |
-
*
|
| 127 |
-
*
|
| 128 |
-
|
| 129 |
-
= 1.
|
| 130 |
-
*
|
| 131 |
-
*
|
| 132 |
-
|
| 133 |
-
= 1.
|
| 134 |
-
*
|
| 135 |
-
|
| 136 |
-
|
| 137 |
-
* Solved
|
| 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 |
-
= 0.
|
| 167 |
-
*
|
| 168 |
-
|
| 169 |
-
|
| 170 |
-
|
| 171 |
-
|
| 172 |
-
*
|
| 173 |
-
|
| 174 |
-
= 0.
|
| 175 |
-
*
|
| 176 |
-
*
|
| 177 |
-
|
| 178 |
-
= 0.
|
| 179 |
-
*
|
| 180 |
-
|
| 181 |
-
|
| 182 |
-
|
| 183 |
-
|
| 184 |
-
*
|
| 185 |
-
|
| 186 |
-
|
| 187 |
-
|
| 188 |
-
|
| 189 |
-
|
| 190 |
-
|
| 191 |
-
|
| 192 |
-
|
| 193 |
-
|
| 194 |
-
*
|
| 195 |
-
*
|
| 196 |
-
|
| 197 |
-
|
| 198 |
-
*
|
| 199 |
-
|
| 200 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
=== WP External Links (nofollow new window seo) ===
|
| 2 |
+
Contributors: freelancephp
|
| 3 |
+
Tags: links, external, icon, target, _blank, _new, _none, rel, nofollow, new window, new tab, javascript, xhtml, seo
|
| 4 |
+
Requires at least: 3.4.0
|
| 5 |
+
Tested up to: 3.8.2
|
| 6 |
+
Stable tag: 1.52
|
| 7 |
+
|
| 8 |
+
Open external links in a new window or tab, adding "nofollow", set link icon, styling, SEO friendly options and more. Easy install and go.
|
| 9 |
+
|
| 10 |
+
== Description ==
|
| 11 |
+
|
| 12 |
+
Configure settings for all external links on your site.
|
| 13 |
+
|
| 14 |
+
= Features =
|
| 15 |
+
* Open in new window or tab
|
| 16 |
+
* Add "nofollow"
|
| 17 |
+
* Choose from 20 icons
|
| 18 |
+
* Set other link options (like classes, title etc)
|
| 19 |
+
* Make it SEO friendly
|
| 20 |
+
|
| 21 |
+
= Easy to use =
|
| 22 |
+
After activating the plugin all options are already set to make your external links SEO friendly. Optionally you can also set the target for opening in a new window or tab or styling options, like adding an icon.
|
| 23 |
+
|
| 24 |
+
= Sources =
|
| 25 |
+
* [Documentation](http://wordpress.org/extend/plugins/wp-external-links/other_notes/)
|
| 26 |
+
* [FAQ](http://wordpress.org/extend/plugins/wp-external-links/faq/)
|
| 27 |
+
* [Github](https://github.com/freelancephp/WP-External-Links)
|
| 28 |
+
|
| 29 |
+
= Like this plugin? =
|
| 30 |
+
[Send your review](http://wordpress.org/support/view/plugin-reviews/wp-external-links-plugin).
|
| 31 |
+
|
| 32 |
+
|
| 33 |
+
== Installation ==
|
| 34 |
+
|
| 35 |
+
1. Go to `Plugins` in the Admin menu
|
| 36 |
+
1. Click on the button `Add new`
|
| 37 |
+
1. Search for `WP External Links` and click 'Install Now' OR click on the `upload` link to upload `wp-external-links.zip`
|
| 38 |
+
1. Click on `Activate plugin`
|
| 39 |
+
|
| 40 |
+
== Frequently Asked Questions ==
|
| 41 |
+
|
| 42 |
+
= I want internal links to be treated as external links. How? =
|
| 43 |
+
|
| 44 |
+
You could add `rel="external"` to those internal links that should be treated as external. The plugin settings will also be applied to those links.
|
| 45 |
+
|
| 46 |
+
= Links to my own domain are treated as external links. Why? =
|
| 47 |
+
|
| 48 |
+
Links pointing to your WordPress site are internal links. All other links will be treated as external links.
|
| 49 |
+
|
| 50 |
+
= I want links to my own domain not being treated as external links. =
|
| 51 |
+
|
| 52 |
+
Add your domain to the option "Ingore links (URL) containing...".
|
| 53 |
+
|
| 54 |
+
[Do you have a question? Please ask me](http://www.freelancephp.net/contact/)
|
| 55 |
+
|
| 56 |
+
== Screenshots ==
|
| 57 |
+
|
| 58 |
+
1. Link Icon on the Site
|
| 59 |
+
1. Admin Settings Page
|
| 60 |
+
|
| 61 |
+
== Documentation ==
|
| 62 |
+
|
| 63 |
+
After activating the plugin all options are already set to make your external links SEO friendly. Optionally you can also set the target for opening in a new window or tab or styling options, like adding an icon.
|
| 64 |
+
|
| 65 |
+
= Action hook =
|
| 66 |
+
The plugin also has a hook when ready, f.e. to add extra filters:
|
| 67 |
+
`function extra_filters($filter_callback, $object) {
|
| 68 |
+
add_filter('some_filter', $filter_callback);
|
| 69 |
+
}
|
| 70 |
+
add_action('wpel_ready', 'extra_filters');`
|
| 71 |
+
|
| 72 |
+
= Filter hook =
|
| 73 |
+
The wpel_external_link filter gives you the possibility to manipulate output of the mailto created by the plugin, like:
|
| 74 |
+
`function special_external_link($created_link, $original_link, $label, $attrs, $is_ignored_link) {
|
| 75 |
+
// skip links that contain the class "not-external"
|
| 76 |
+
if (isset($attrs['class']) && strpos($attrs['class'], 'not-external') !== false) {
|
| 77 |
+
return $original_link;
|
| 78 |
+
}
|
| 79 |
+
|
| 80 |
+
return '<b>'. $created_link .'</b>';
|
| 81 |
+
}
|
| 82 |
+
add_filter('wpel_external_link', 'special_external_link', 10, 5);`
|
| 83 |
+
|
| 84 |
+
Now all external links will be processed and wrapped around a `<b>`-tag. And links containing the class "not-external" will not be processed by the plugin at all (and stay the way they are).
|
| 85 |
+
|
| 86 |
+
= Credits =
|
| 87 |
+
* [jQuery Tipsy Plugin](http://plugins.jquery.com/project/tipsy) made by [Jason Frame](http://onehackoranother.com/)
|
| 88 |
+
* [phpQuery](http://code.google.com/p/phpquery/) made by [Tobiasz Cudnik](http://tobiasz123.wordpress.com)
|
| 89 |
+
* [Icon](http://findicons.com/icon/164579/link_go?id=427009) made by [FatCow Web Hosting](http://www.fatcow.com/)
|
| 90 |
+
|
| 91 |
+
== Changelog ==
|
| 92 |
+
|
| 93 |
+
= 1.52 =
|
| 94 |
+
* Added filter hook wpel_internal_link
|
| 95 |
+
* Fixed use_js option bug
|
| 96 |
+
* Fixed bug loading non-existing stylesheet
|
| 97 |
+
* Minified javascripts
|
| 98 |
+
|
| 99 |
+
= 1.51 =
|
| 100 |
+
* Fixed also check url's starting with //
|
| 101 |
+
* Fixed wpel_external_link also applied on ignored links
|
| 102 |
+
|
| 103 |
+
= 1.50 =
|
| 104 |
+
* Removed stylesheet file to save extra request
|
| 105 |
+
* Added option for loading js file in wp_footer
|
| 106 |
+
* Fixed bug with data-* attributes
|
| 107 |
+
* Fixed bug url's with hash at the end
|
| 108 |
+
* Fixed PHP errors
|
| 109 |
+
|
| 110 |
+
= 1.41 =
|
| 111 |
+
* Fixed Bug: wpmel_external_link filter hook was not working correctly
|
| 112 |
+
|
| 113 |
+
= 1.40 =
|
| 114 |
+
* Added action hook wpel_ready
|
| 115 |
+
* Added filter hook wpel_external_link
|
| 116 |
+
* Added output flush on wp_footer
|
| 117 |
+
* Fixed Bug: spaces before url in href-attribute not recognized as external link
|
| 118 |
+
* Fixed Bug: external links not processed (regexpr tag conflict starting with an a, like <aside> or <article>)
|
| 119 |
+
* Cosmetic changes: added "Admin Settings", replaced help icon, restyled tooltip texts, removed "About this plugin" box
|
| 120 |
+
|
| 121 |
+
= 1.31 =
|
| 122 |
+
* Fixed passing arguments by reference using & (deprecated for PHP 5.4+)
|
| 123 |
+
* Fixed options save failure by adding a non-ajax submit fallback
|
| 124 |
+
|
| 125 |
+
= 1.30 =
|
| 126 |
+
* Re-arranged options in metaboxes
|
| 127 |
+
* Added option for no icons on images
|
| 128 |
+
|
| 129 |
+
= 1.21 =
|
| 130 |
+
* Fixed phpQuery bugs (class already exists and loading stylesheet)
|
| 131 |
+
* Solved php notices
|
| 132 |
+
|
| 133 |
+
= 1.20 =
|
| 134 |
+
* Added option to ignore certain links or domains
|
| 135 |
+
* Solved tweet button problem by adding link to new ignore option
|
| 136 |
+
* Made JavaScript method consistent to not using JS
|
| 137 |
+
* Solved PHP warnings
|
| 138 |
+
* Solved bug adding own class
|
| 139 |
+
* Changed bloginfo "url" to "wpurl"
|
| 140 |
+
|
| 141 |
+
= 1.10 =
|
| 142 |
+
* Resolved old parsing method (same as version 0.35)
|
| 143 |
+
* Option to use phpQuery for parsing (for those who didn't experience problems with version 1.03)
|
| 144 |
+
|
| 145 |
+
= 1.03 =
|
| 146 |
+
* Workaround for echo DOCTYPE bug (caused by attributes in the head-tag)
|
| 147 |
+
|
| 148 |
+
= 1.02 =
|
| 149 |
+
* Solved the not working activation hook
|
| 150 |
+
|
| 151 |
+
= 1.01 =
|
| 152 |
+
* Solved bug after live testing
|
| 153 |
+
|
| 154 |
+
= 1.00 =
|
| 155 |
+
* Added option for setting title-attribute
|
| 156 |
+
* Added option for excluding filtering certain external links
|
| 157 |
+
* Added Admin help tooltips using jQuery Tipsy Plugin
|
| 158 |
+
* Reorginized files and refactored code to PHP5 (no support for PHP4)
|
| 159 |
+
* Added WP built-in meta box functionallity (using the `WP_Meta_Box_Page` Class)
|
| 160 |
+
* Reorganized saving options and added Ajax save method (using the `WP_Option_Forms` Class)
|
| 161 |
+
* Removed Regexp and using phpQuery
|
| 162 |
+
* Choose menu position for this plugin (see "Screen Options")
|
| 163 |
+
* Removed possibility to convert all `<a>` tags to xhtml clean code (so only external links will be converted)
|
| 164 |
+
* Removed "Solve problem" options
|
| 165 |
+
|
| 166 |
+
= 0.35 =
|
| 167 |
+
* Widget Logic options bug
|
| 168 |
+
|
| 169 |
+
= 0.34 =
|
| 170 |
+
* Added option only converting external `<a>` tags to XHTML valid code
|
| 171 |
+
* Changed script attribute `language` to `type`
|
| 172 |
+
* Added support for widget_content filter of the Logic Widget plugin
|
| 173 |
+
|
| 174 |
+
= 0.33 =
|
| 175 |
+
* Added option to fix js problem
|
| 176 |
+
* Fixed PHP / WP notices
|
| 177 |
+
|
| 178 |
+
= 0.32 =
|
| 179 |
+
* For jQuery uses live() function so also opens dynamicly created links in given target
|
| 180 |
+
* Fixed bug of changing `<abbr>` tag
|
| 181 |
+
* Small cosmetical adjustments
|
| 182 |
+
|
| 183 |
+
= 0.31 =
|
| 184 |
+
* Small cosmetical adjustments
|
| 185 |
+
|
| 186 |
+
= 0.30 =
|
| 187 |
+
* Improved Admin Options, f.e. target option looks more like the Blogroll target option
|
| 188 |
+
* Added option for choosing which content should be filtered
|
| 189 |
+
|
| 190 |
+
= 0.21 =
|
| 191 |
+
* Solved bug removing icon stylesheet
|
| 192 |
+
|
| 193 |
+
= 0.20 =
|
| 194 |
+
* Put icon styles in external stylesheet
|
| 195 |
+
* Can use "ext-icon-..." to show a specific icon on a link
|
| 196 |
+
* Added option to set your own No-Icon class
|
| 197 |
+
* Made "Class" optional, so it's not used for showing icons anymore
|
| 198 |
+
* Added 3 more icons
|
| 199 |
+
|
| 200 |
+
= 0.12 =
|
| 201 |
+
* Options are organized more logical
|
| 202 |
+
* Added some more icons
|
| 203 |
+
|
| 204 |
+
= 0.11 =
|
| 205 |
+
* JavaScript uses window.open() (tested in FireFox Opera, Safari, Chrome and IE6+)
|
| 206 |
+
* Also possible to open all external links in the same new window
|
| 207 |
+
* Some layout changes on the Admin Options Page
|
| 208 |
+
|
| 209 |
+
= 0.10 =
|
| 210 |
+
* Features: opening in a new window, set link icon, set "external", set "nofollow", set css-class
|
| 211 |
+
* Replaces external links by clean XHTML <a> tags
|
| 212 |
+
* Internalization implemented (no language files yet)
|
wp-external-links.php
CHANGED
|
@@ -1,53 +1,53 @@
|
|
| 1 |
-
<?php defined('ABSPATH') OR die('No direct access.');
|
| 2 |
-
/*
|
| 3 |
-
Plugin Name: WP External Links
|
| 4 |
-
Plugin URI: http://www.freelancephp.net/wp-external-links-plugin
|
| 5 |
-
Description: Open external links in a new window/tab, add "external" / "nofollow" to rel-attribute, set icon, XHTML strict, SEO friendly...
|
| 6 |
-
Author: Victor Villaverde Laan
|
| 7 |
-
Version: 1.
|
| 8 |
-
Author URI: http://www.freelancephp.net
|
| 9 |
-
License: Dual licensed under the MIT and GPL licenses
|
| 10 |
-
*/
|
| 11 |
-
|
| 12 |
-
// constants
|
| 13 |
-
if (!defined('WP_EXTERNAL_LINKS_FILE')) { define('WP_EXTERNAL_LINKS_FILE', __FILE__); }
|
| 14 |
-
if (!defined('WP_EXTERNAL_LINKS_VERSION')) { define('WP_EXTERNAL_LINKS_VERSION', '1.
|
| 15 |
-
if (!defined('WP_EXTERNAL_LINKS_KEY')) { define('WP_EXTERNAL_LINKS_KEY', 'wp_external_links'); }
|
| 16 |
-
if (!defined('WP_EXTERNAL_LINKS_DOMAIN')) { define('WP_EXTERNAL_LINKS_DOMAIN', 'wp-external-links'); }
|
| 17 |
-
if (!defined('WP_EXTERNAL_LINKS_OPTIONS_NAME')) { define('WP_EXTERNAL_LINKS_OPTIONS_NAME', 'WP_External_Links_options'); }
|
| 18 |
-
if (!defined('WP_EXTERNAL_LINKS_ADMIN_PAGE')) { define('WP_EXTERNAL_LINKS_ADMIN_PAGE', 'wp-external-links-settings'); }
|
| 19 |
-
|
| 20 |
-
// check plugin compatibility
|
| 21 |
-
if (isset($wp_version)
|
| 22 |
-
AND version_compare(preg_replace('/-.*$/', '', $wp_version), '3.4', '>=')
|
| 23 |
-
AND version_compare(phpversion(), '5.2.4', '>=')) {
|
| 24 |
-
|
| 25 |
-
// include classes
|
| 26 |
-
require_once('includes/wp-plugin-dev-classes/class-wp-meta-box-page.php');
|
| 27 |
-
require_once('includes/wp-plugin-dev-classes/class-wp-option-forms.php');
|
| 28 |
-
require_once('includes/class-admin-external-links.php');
|
| 29 |
-
require_once('includes/class-wp-external-links.php');
|
| 30 |
-
|
| 31 |
-
// create instance
|
| 32 |
-
$WP_External_Links = new WP_External_Links();
|
| 33 |
-
|
| 34 |
-
} else {
|
| 35 |
-
|
| 36 |
-
// set error message
|
| 37 |
-
if (!function_exists('wpel_error_notice')):
|
| 38 |
-
function wpel_error_notice() {
|
| 39 |
-
$plugin_title = get_admin_page_title();
|
| 40 |
-
|
| 41 |
-
echo '<div class="error">'
|
| 42 |
-
. sprintf(__('<p>Warning - The plugin <strong>%s</strong> requires PHP 5.2.4+ and WP 3.4+. Please upgrade your PHP and/or WordPress.'
|
| 43 |
-
. '<br/>Disable the plugin to remove this message.</p>'
|
| 44 |
-
, WP_EXTERNAL_LINKS_KEY), $plugin_title)
|
| 45 |
-
. '</div>';
|
| 46 |
-
}
|
| 47 |
-
|
| 48 |
-
add_action('admin_notices', 'wpel_error_notice');
|
| 49 |
-
endif;
|
| 50 |
-
|
| 51 |
-
}
|
| 52 |
-
|
| 53 |
/* ommit PHP closing tag, to prevent unwanted whitespace at the end of the parts generated by the included files */
|
| 1 |
+
<?php defined('ABSPATH') OR die('No direct access.');
|
| 2 |
+
/*
|
| 3 |
+
Plugin Name: WP External Links
|
| 4 |
+
Plugin URI: http://www.freelancephp.net/wp-external-links-plugin
|
| 5 |
+
Description: Open external links in a new window/tab, add "external" / "nofollow" to rel-attribute, set icon, XHTML strict, SEO friendly...
|
| 6 |
+
Author: Victor Villaverde Laan
|
| 7 |
+
Version: 1.52
|
| 8 |
+
Author URI: http://www.freelancephp.net
|
| 9 |
+
License: Dual licensed under the MIT and GPL licenses
|
| 10 |
+
*/
|
| 11 |
+
|
| 12 |
+
// constants
|
| 13 |
+
if (!defined('WP_EXTERNAL_LINKS_FILE')) { define('WP_EXTERNAL_LINKS_FILE', __FILE__); }
|
| 14 |
+
if (!defined('WP_EXTERNAL_LINKS_VERSION')) { define('WP_EXTERNAL_LINKS_VERSION', '1.52'); }
|
| 15 |
+
if (!defined('WP_EXTERNAL_LINKS_KEY')) { define('WP_EXTERNAL_LINKS_KEY', 'wp_external_links'); }
|
| 16 |
+
if (!defined('WP_EXTERNAL_LINKS_DOMAIN')) { define('WP_EXTERNAL_LINKS_DOMAIN', 'wp-external-links'); }
|
| 17 |
+
if (!defined('WP_EXTERNAL_LINKS_OPTIONS_NAME')) { define('WP_EXTERNAL_LINKS_OPTIONS_NAME', 'WP_External_Links_options'); }
|
| 18 |
+
if (!defined('WP_EXTERNAL_LINKS_ADMIN_PAGE')) { define('WP_EXTERNAL_LINKS_ADMIN_PAGE', 'wp-external-links-settings'); }
|
| 19 |
+
|
| 20 |
+
// check plugin compatibility
|
| 21 |
+
if (isset($wp_version)
|
| 22 |
+
AND version_compare(preg_replace('/-.*$/', '', $wp_version), '3.4', '>=')
|
| 23 |
+
AND version_compare(phpversion(), '5.2.4', '>=')) {
|
| 24 |
+
|
| 25 |
+
// include classes
|
| 26 |
+
require_once('includes/wp-plugin-dev-classes/class-wp-meta-box-page.php');
|
| 27 |
+
require_once('includes/wp-plugin-dev-classes/class-wp-option-forms.php');
|
| 28 |
+
require_once('includes/class-admin-external-links.php');
|
| 29 |
+
require_once('includes/class-wp-external-links.php');
|
| 30 |
+
|
| 31 |
+
// create instance
|
| 32 |
+
$WP_External_Links = new WP_External_Links();
|
| 33 |
+
|
| 34 |
+
} else {
|
| 35 |
+
|
| 36 |
+
// set error message
|
| 37 |
+
if (!function_exists('wpel_error_notice')):
|
| 38 |
+
function wpel_error_notice() {
|
| 39 |
+
$plugin_title = get_admin_page_title();
|
| 40 |
+
|
| 41 |
+
echo '<div class="error">'
|
| 42 |
+
. sprintf(__('<p>Warning - The plugin <strong>%s</strong> requires PHP 5.2.4+ and WP 3.4+. Please upgrade your PHP and/or WordPress.'
|
| 43 |
+
. '<br/>Disable the plugin to remove this message.</p>'
|
| 44 |
+
, WP_EXTERNAL_LINKS_KEY), $plugin_title)
|
| 45 |
+
. '</div>';
|
| 46 |
+
}
|
| 47 |
+
|
| 48 |
+
add_action('admin_notices', 'wpel_error_notice');
|
| 49 |
+
endif;
|
| 50 |
+
|
| 51 |
+
}
|
| 52 |
+
|
| 53 |
/* ommit PHP closing tag, to prevent unwanted whitespace at the end of the parts generated by the included files */
|
