phpQuery.php 42 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344
  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. // class names for instanceof
  15. // TODO move them as class constants into phpQuery
  16. define('DOMDOCUMENT', 'DOMDocument');
  17. define('DOMELEMENT', 'DOMElement');
  18. define('DOMNODELIST', 'DOMNodeList');
  19. define('DOMNODE', 'DOMNode');
  20. require_once(dirname(__FILE__).'/phpQuery/DOMEvent.php');
  21. require_once(dirname(__FILE__).'/phpQuery/DOMDocumentWrapper.php');
  22. require_once(dirname(__FILE__).'/phpQuery/phpQueryEvents.php');
  23. require_once(dirname(__FILE__).'/phpQuery/Callback.php');
  24. require_once(dirname(__FILE__).'/phpQuery/phpQueryObject.php');
  25. require_once(dirname(__FILE__).'/phpQuery/compat/mbstring.php');
  26. /**
  27. * Static namespace for phpQuery functions.
  28. *
  29. * @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
  30. * @package phpQuery
  31. */
  32. abstract class phpQuery {
  33. /**
  34. * XXX: Workaround for mbstring problems
  35. *
  36. * @var bool
  37. */
  38. public static $mbstringSupport = true;
  39. public static $debug = false;
  40. public static $documents = array();
  41. public static $defaultDocumentID = null;
  42. // public static $defaultDoctype = 'html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"';
  43. /**
  44. * Applies only to HTML.
  45. *
  46. * @var unknown_type
  47. */
  48. public static $defaultDoctype = '<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"
  49. "http://www.w3.org/TR/html4/loose.dtd">';
  50. public static $defaultCharset = 'UTF-8';
  51. /**
  52. * Static namespace for plugins.
  53. *
  54. * @var object
  55. */
  56. public static $plugins = array();
  57. /**
  58. * List of loaded plugins.
  59. *
  60. * @var unknown_type
  61. */
  62. public static $pluginsLoaded = array();
  63. public static $pluginsMethods = array();
  64. public static $pluginsStaticMethods = array();
  65. public static $extendMethods = array();
  66. /**
  67. * @TODO implement
  68. */
  69. public static $extendStaticMethods = array();
  70. /**
  71. * Hosts allowed for AJAX connections.
  72. * Dot '.' means $_SERVER['HTTP_HOST'] (if any).
  73. *
  74. * @var array
  75. */
  76. public static $ajaxAllowedHosts = array(
  77. '.'
  78. );
  79. /**
  80. * AJAX settings.
  81. *
  82. * @var array
  83. * XXX should it be static or not ?
  84. */
  85. public static $ajaxSettings = array(
  86. 'url' => '',//TODO
  87. 'global' => true,
  88. 'type' => "GET",
  89. 'timeout' => null,
  90. 'contentType' => "application/x-www-form-urlencoded",
  91. 'processData' => true,
  92. // 'async' => true,
  93. 'data' => null,
  94. 'username' => null,
  95. 'password' => null,
  96. 'dataType' => null,
  97. 'ifModified' => null,
  98. 'accepts' => array(
  99. 'xml' => "application/xml, text/xml",
  100. 'html' => "text/html",
  101. 'script' => "text/javascript, application/javascript",
  102. 'json' => "application/json, text/javascript",
  103. 'text' => "text/plain",
  104. '_default' => "*/*"
  105. )
  106. );
  107. public static $lastModified = null;
  108. public static $active = 0;
  109. public static $dumpCount = 0;
  110. /**
  111. * Multi-purpose function.
  112. * Use pq() as shortcut.
  113. *
  114. * In below examples, $pq is any result of pq(); function.
  115. *
  116. * 1. Import markup into existing document (without any attaching):
  117. * - Import into selected document:
  118. * pq('<div/>') // DOESNT accept text nodes at beginning of input string !
  119. * - Import into document with ID from $pq->getDocumentID():
  120. * pq('<div/>', $pq->getDocumentID())
  121. * - Import into same document as DOMNode belongs to:
  122. * pq('<div/>', DOMNode)
  123. * - Import into document from phpQuery object:
  124. * pq('<div/>', $pq)
  125. *
  126. * 2. Run query:
  127. * - Run query on last selected document:
  128. * pq('div.myClass')
  129. * - Run query on document with ID from $pq->getDocumentID():
  130. * pq('div.myClass', $pq->getDocumentID())
  131. * - Run query on same document as DOMNode belongs to and use node(s)as root for query:
  132. * pq('div.myClass', DOMNode)
  133. * - Run query on document from phpQuery object
  134. * and use object's stack as root node(s) for query:
  135. * pq('div.myClass', $pq)
  136. *
  137. * @param string|DOMNode|DOMNodeList|array $arg1 HTML markup, CSS Selector, DOMNode or array of DOMNodes
  138. * @param string|phpQueryObject|DOMNode $context DOM ID from $pq->getDocumentID(), phpQuery object (determines also query root) or DOMNode (determines also query root)
  139. *
  140. * @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery|QueryTemplatesPhpQuery|false
  141. * phpQuery object or false in case of error.
  142. */
  143. public static function pq($arg1, $context = null) {
  144. if ($arg1 instanceof DOMNODE && ! isset($context)) {
  145. foreach(phpQuery::$documents as $documentWrapper) {
  146. $compare = $arg1 instanceof DOMDocument
  147. ? $arg1 : $arg1->ownerDocument;
  148. if ($documentWrapper->document->isSameNode($compare))
  149. $context = $documentWrapper->id;
  150. }
  151. }
  152. if (! $context) {
  153. $domId = self::$defaultDocumentID;
  154. if (! $domId)
  155. throw new Exception("Can't use last created DOM, because there isn't any. Use phpQuery::newDocument() first.");
  156. // } else if (is_object($context) && ($context instanceof PHPQUERY || is_subclass_of($context, 'phpQueryObject')))
  157. } else if (is_object($context) && $context instanceof phpQueryObject)
  158. $domId = $context->getDocumentID();
  159. else if ($context instanceof DOMDOCUMENT) {
  160. $domId = self::getDocumentID($context);
  161. if (! $domId) {
  162. //throw new Exception('Orphaned DOMDocument');
  163. $domId = self::newDocument($context)->getDocumentID();
  164. }
  165. } else if ($context instanceof DOMNODE) {
  166. $domId = self::getDocumentID($context);
  167. if (! $domId) {
  168. throw new Exception('Orphaned DOMNode');
  169. // $domId = self::newDocument($context->ownerDocument);
  170. }
  171. } else
  172. $domId = $context;
  173. if ($arg1 instanceof phpQueryObject) {
  174. // if (is_object($arg1) && (get_class($arg1) == 'phpQueryObject' || $arg1 instanceof PHPQUERY || is_subclass_of($arg1, 'phpQueryObject'))) {
  175. /**
  176. * Return $arg1 or import $arg1 stack if document differs:
  177. * pq(pq('<div/>'))
  178. */
  179. if ($arg1->getDocumentID() == $domId)
  180. return $arg1;
  181. $class = get_class($arg1);
  182. // support inheritance by passing old object to overloaded constructor
  183. $phpQuery = $class != 'phpQuery'
  184. ? new $class($arg1, $domId)
  185. : new phpQueryObject($domId);
  186. $phpQuery->elements = array();
  187. foreach($arg1->elements as $node)
  188. $phpQuery->elements[] = $phpQuery->document->importNode($node, true);
  189. return $phpQuery;
  190. } else if ($arg1 instanceof DOMNODE || (is_array($arg1) && isset($arg1[0]) && $arg1[0] instanceof DOMNODE)) {
  191. /*
  192. * Wrap DOM nodes with phpQuery object, import into document when needed:
  193. * pq(array($domNode1, $domNode2))
  194. */
  195. $phpQuery = new phpQueryObject($domId);
  196. if (!($arg1 instanceof DOMNODELIST) && ! is_array($arg1))
  197. $arg1 = array($arg1);
  198. $phpQuery->elements = array();
  199. foreach($arg1 as $node) {
  200. $sameDocument = $node->ownerDocument instanceof DOMDOCUMENT
  201. && ! $node->ownerDocument->isSameNode($phpQuery->document);
  202. $phpQuery->elements[] = $sameDocument
  203. ? $phpQuery->document->importNode($node, true)
  204. : $node;
  205. }
  206. return $phpQuery;
  207. } else if (self::isMarkup($arg1)) {
  208. /**
  209. * Import HTML:
  210. * pq('<div/>')
  211. */
  212. $phpQuery = new phpQueryObject($domId);
  213. return $phpQuery->newInstance(
  214. $phpQuery->documentWrapper->import($arg1)
  215. );
  216. } else {
  217. /**
  218. * Run CSS query:
  219. * pq('div.myClass')
  220. */
  221. $phpQuery = new phpQueryObject($domId);
  222. // if ($context && ($context instanceof PHPQUERY || is_subclass_of($context, 'phpQueryObject')))
  223. if ($context && $context instanceof phpQueryObject)
  224. $phpQuery->elements = $context->elements;
  225. else if ($context && $context instanceof DOMNODELIST) {
  226. $phpQuery->elements = array();
  227. foreach($context as $node)
  228. $phpQuery->elements[] = $node;
  229. } else if ($context && $context instanceof DOMNODE)
  230. $phpQuery->elements = array($context);
  231. return $phpQuery->find($arg1);
  232. }
  233. }
  234. /**
  235. * Sets default document to $id. Document has to be loaded prior
  236. * to using this method.
  237. * $id can be retrived via getDocumentID() or getDocumentIDRef().
  238. *
  239. * @param unknown_type $id
  240. */
  241. public static function selectDocument($id) {
  242. $id = self::getDocumentID($id);
  243. self::debug("Selecting document '$id' as default one");
  244. self::$defaultDocumentID = self::getDocumentID($id);
  245. }
  246. /**
  247. * Returns document with id $id or last used as phpQueryObject.
  248. * $id can be retrived via getDocumentID() or getDocumentIDRef().
  249. * Chainable.
  250. *
  251. * @see phpQuery::selectDocument()
  252. * @param unknown_type $id
  253. * @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
  254. */
  255. public static function getDocument($id = null) {
  256. if ($id)
  257. phpQuery::selectDocument($id);
  258. else
  259. $id = phpQuery::$defaultDocumentID;
  260. return new phpQueryObject($id);
  261. }
  262. /**
  263. * Creates new document from markup.
  264. * Chainable.
  265. *
  266. * @param unknown_type $markup
  267. * @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
  268. */
  269. public static function newDocument($markup = null, $contentType = null) {
  270. if (! $markup)
  271. $markup = '';
  272. $documentID = phpQuery::createDocumentWrapper($markup, $contentType);
  273. return new phpQueryObject($documentID);
  274. }
  275. /**
  276. * Creates new document from markup.
  277. * Chainable.
  278. *
  279. * @param unknown_type $markup
  280. * @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
  281. */
  282. public static function newDocumentHTML($markup = null, $charset = null) {
  283. $contentType = $charset
  284. ? ";charset=$charset"
  285. : '';
  286. return self::newDocument($markup, "text/html{$contentType}");
  287. }
  288. /**
  289. * Creates new document from markup.
  290. * Chainable.
  291. *
  292. * @param unknown_type $markup
  293. * @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
  294. */
  295. public static function newDocumentXML($markup = null, $charset = null) {
  296. $contentType = $charset
  297. ? ";charset=$charset"
  298. : '';
  299. return self::newDocument($markup, "text/xml{$contentType}");
  300. }
  301. /**
  302. * Creates new document from markup.
  303. * Chainable.
  304. *
  305. * @param unknown_type $markup
  306. * @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
  307. */
  308. public static function newDocumentXHTML($markup = null, $charset = null) {
  309. $contentType = $charset
  310. ? ";charset=$charset"
  311. : '';
  312. return self::newDocument($markup, "application/xhtml+xml{$contentType}");
  313. }
  314. /**
  315. * Creates new document from markup.
  316. * Chainable.
  317. *
  318. * @param unknown_type $markup
  319. * @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
  320. */
  321. public static function newDocumentPHP($markup = null, $contentType = "text/html") {
  322. // TODO pass charset to phpToMarkup if possible (use DOMDocumentWrapper function)
  323. $markup = phpQuery::phpToMarkup($markup, self::$defaultCharset);
  324. return self::newDocument($markup, $contentType);
  325. }
  326. public static function phpToMarkup($php, $charset = 'utf-8') {
  327. $regexes = array(
  328. '@(<(?!\\?)(?:[^>]|\\?>)+\\w+\\s*=\\s*)(\')([^\']*)<'.'?php?(.*?)(?:\\?>)([^\']*)\'@s',
  329. '@(<(?!\\?)(?:[^>]|\\?>)+\\w+\\s*=\\s*)(")([^"]*)<'.'?php?(.*?)(?:\\?>)([^"]*)"@s',
  330. );
  331. foreach($regexes as $regex)
  332. while (preg_match($regex, $php, $matches)) {
  333. $php = preg_replace_callback(
  334. $regex,
  335. // create_function('$m, $charset = "'.$charset.'"',
  336. // 'return $m[1].$m[2]
  337. // .htmlspecialchars("<"."?php".$m[4]."?".">", ENT_QUOTES|ENT_NOQUOTES, $charset)
  338. // .$m[5].$m[2];'
  339. // ),
  340. array('phpQuery', '_phpToMarkupCallback'),
  341. $php
  342. );
  343. }
  344. $regex = '@(^|>[^<]*)+?(<\?php(.*?)(\?>))@s';
  345. //preg_match_all($regex, $php, $matches);
  346. //var_dump($matches);
  347. $php = preg_replace($regex, '\\1<php><!-- \\3 --></php>', $php);
  348. return $php;
  349. }
  350. public static function _phpToMarkupCallback($php, $charset = 'utf-8') {
  351. return $m[1].$m[2]
  352. .htmlspecialchars("<"."?php".$m[4]."?".">", ENT_QUOTES|ENT_NOQUOTES, $charset)
  353. .$m[5].$m[2];
  354. }
  355. public static function _markupToPHPCallback($m) {
  356. return "<"."?php ".htmlspecialchars_decode($m[1])." ?".">";
  357. }
  358. /**
  359. * Converts document markup containing PHP code generated by phpQuery::php()
  360. * into valid (executable) PHP code syntax.
  361. *
  362. * @param string|phpQueryObject $content
  363. * @return string PHP code.
  364. */
  365. public static function markupToPHP($content) {
  366. if ($content instanceof phpQueryObject)
  367. $content = $content->markupOuter();
  368. /* <php>...</php> to <?php...? > */
  369. $content = preg_replace_callback(
  370. '@<php>\s*<!--(.*?)-->\s*</php>@s',
  371. // create_function('$m',
  372. // 'return "<'.'?php ".htmlspecialchars_decode($m[1])." ?'.'>";'
  373. // ),
  374. array('phpQuery', '_markupToPHPCallback'),
  375. $content
  376. );
  377. /* <node attr='< ?php ? >'> extra space added to save highlighters */
  378. $regexes = array(
  379. '@(<(?!\\?)(?:[^>]|\\?>)+\\w+\\s*=\\s*)(\')([^\']*)(?:&lt;|%3C)\\?(?:php)?(.*?)(?:\\?(?:&gt;|%3E))([^\']*)\'@s',
  380. '@(<(?!\\?)(?:[^>]|\\?>)+\\w+\\s*=\\s*)(")([^"]*)(?:&lt;|%3C)\\?(?:php)?(.*?)(?:\\?(?:&gt;|%3E))([^"]*)"@s',
  381. );
  382. foreach($regexes as $regex)
  383. while (preg_match($regex, $content))
  384. $content = preg_replace_callback(
  385. $regex,
  386. create_function('$m',
  387. 'return $m[1].$m[2].$m[3]."<?php "
  388. .str_replace(
  389. array("%20", "%3E", "%09", "&#10;", "&#9;", "%7B", "%24", "%7D", "%22", "%5B", "%5D"),
  390. array(" ", ">", " ", "\n", " ", "{", "$", "}", \'"\', "[", "]"),
  391. htmlspecialchars_decode($m[4])
  392. )
  393. ." ?>".$m[5].$m[2];'
  394. ),
  395. $content
  396. );
  397. return $content;
  398. }
  399. /**
  400. * Creates new document from file $file.
  401. * Chainable.
  402. *
  403. * @param string $file URLs allowed. See File wrapper page at php.net for more supported sources.
  404. * @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
  405. */
  406. public static function newDocumentFile($file, $contentType = null) {
  407. $documentID = self::createDocumentWrapper(
  408. file_get_contents($file), $contentType
  409. );
  410. return new phpQueryObject($documentID);
  411. }
  412. /**
  413. * Creates new document from markup.
  414. * Chainable.
  415. *
  416. * @param unknown_type $markup
  417. * @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
  418. */
  419. public static function newDocumentFileHTML($file, $charset = null) {
  420. $contentType = $charset
  421. ? ";charset=$charset"
  422. : '';
  423. return self::newDocumentFile($file, "text/html{$contentType}");
  424. }
  425. /**
  426. * Creates new document from markup.
  427. * Chainable.
  428. *
  429. * @param unknown_type $markup
  430. * @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
  431. */
  432. public static function newDocumentFileXML($file, $charset = null) {
  433. $contentType = $charset
  434. ? ";charset=$charset"
  435. : '';
  436. return self::newDocumentFile($file, "text/xml{$contentType}");
  437. }
  438. /**
  439. * Creates new document from markup.
  440. * Chainable.
  441. *
  442. * @param unknown_type $markup
  443. * @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
  444. */
  445. public static function newDocumentFileXHTML($file, $charset = null) {
  446. $contentType = $charset
  447. ? ";charset=$charset"
  448. : '';
  449. return self::newDocumentFile($file, "application/xhtml+xml{$contentType}");
  450. }
  451. /**
  452. * Creates new document from markup.
  453. * Chainable.
  454. *
  455. * @param unknown_type $markup
  456. * @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
  457. */
  458. public static function newDocumentFilePHP($file, $contentType = null) {
  459. return self::newDocumentPHP(file_get_contents($file), $contentType);
  460. }
  461. /**
  462. * Reuses existing DOMDocument object.
  463. * Chainable.
  464. *
  465. * @param $document DOMDocument
  466. * @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
  467. * @TODO support DOMDocument
  468. */
  469. public static function loadDocument($document) {
  470. // TODO
  471. die('TODO loadDocument');
  472. }
  473. /**
  474. * Enter description here...
  475. *
  476. * @param unknown_type $html
  477. * @param unknown_type $domId
  478. * @return unknown New DOM ID
  479. * @todo support PHP tags in input
  480. * @todo support passing DOMDocument object from self::loadDocument
  481. */
  482. protected static function createDocumentWrapper($html, $contentType = null, $documentID = null) {
  483. if (function_exists('domxml_open_mem'))
  484. throw new Exception("Old PHP4 DOM XML extension detected. phpQuery won't work until this extension is enabled.");
  485. // $id = $documentID
  486. // ? $documentID
  487. // : md5(microtime());
  488. $document = null;
  489. if ($html instanceof DOMDOCUMENT) {
  490. if (self::getDocumentID($html)) {
  491. // document already exists in phpQuery::$documents, make a copy
  492. $document = clone $html;
  493. } else {
  494. // new document, add it to phpQuery::$documents
  495. $wrapper = new DOMDocumentWrapper($html, $contentType, $documentID);
  496. }
  497. } else {
  498. $wrapper = new DOMDocumentWrapper($html, $contentType, $documentID);
  499. }
  500. // $wrapper->id = $id;
  501. // bind document
  502. phpQuery::$documents[$wrapper->id] = $wrapper;
  503. // remember last loaded document
  504. phpQuery::selectDocument($wrapper->id);
  505. return $wrapper->id;
  506. }
  507. /**
  508. * Extend class namespace.
  509. *
  510. * @param string|array $target
  511. * @param array $source
  512. * @TODO support string $source
  513. * @return unknown_type
  514. */
  515. public static function extend($target, $source) {
  516. switch($target) {
  517. case 'phpQueryObject':
  518. $targetRef = &self::$extendMethods;
  519. $targetRef2 = &self::$pluginsMethods;
  520. break;
  521. case 'phpQuery':
  522. $targetRef = &self::$extendStaticMethods;
  523. $targetRef2 = &self::$pluginsStaticMethods;
  524. break;
  525. default:
  526. throw new Exception("Unsupported \$target type");
  527. }
  528. if (is_string($source))
  529. $source = array($source => $source);
  530. foreach($source as $method => $callback) {
  531. if (isset($targetRef[$method])) {
  532. // throw new Exception
  533. self::debug("Duplicate method '{$method}', can\'t extend '{$target}'");
  534. continue;
  535. }
  536. if (isset($targetRef2[$method])) {
  537. // throw new Exception
  538. self::debug("Duplicate method '{$method}' from plugin '{$targetRef2[$method]}',"
  539. ." can\'t extend '{$target}'");
  540. continue;
  541. }
  542. $targetRef[$method] = $callback;
  543. }
  544. return true;
  545. }
  546. /**
  547. * Extend phpQuery with $class from $file.
  548. *
  549. * @param string $class Extending class name. Real class name can be prepended phpQuery_.
  550. * @param string $file Filename to include. Defaults to "{$class}.php".
  551. */
  552. public static function plugin($class, $file = null) {
  553. // TODO $class checked agains phpQuery_$class
  554. // if (strpos($class, 'phpQuery') === 0)
  555. // $class = substr($class, 8);
  556. if (in_array($class, self::$pluginsLoaded))
  557. return true;
  558. if (! $file)
  559. $file = $class.'.php';
  560. $objectClassExists = class_exists('phpQueryObjectPlugin_'.$class);
  561. $staticClassExists = class_exists('phpQueryPlugin_'.$class);
  562. if (! $objectClassExists && ! $staticClassExists)
  563. require_once($file);
  564. self::$pluginsLoaded[] = $class;
  565. // static methods
  566. if (class_exists('phpQueryPlugin_'.$class)) {
  567. $realClass = 'phpQueryPlugin_'.$class;
  568. $vars = get_class_vars($realClass);
  569. $loop = isset($vars['phpQueryMethods'])
  570. && ! is_null($vars['phpQueryMethods'])
  571. ? $vars['phpQueryMethods']
  572. : get_class_methods($realClass);
  573. foreach($loop as $method) {
  574. if ($method == '__initialize')
  575. continue;
  576. if (! is_callable(array($realClass, $method)))
  577. continue;
  578. if (isset(self::$pluginsStaticMethods[$method])) {
  579. throw new Exception("Duplicate method '{$method}' from plugin '{$c}' conflicts with same method from plugin '".self::$pluginsStaticMethods[$method]."'");
  580. return;
  581. }
  582. self::$pluginsStaticMethods[$method] = $class;
  583. }
  584. if (method_exists($realClass, '__initialize'))
  585. call_user_func_array(array($realClass, '__initialize'), array());
  586. }
  587. // object methods
  588. if (class_exists('phpQueryObjectPlugin_'.$class)) {
  589. $realClass = 'phpQueryObjectPlugin_'.$class;
  590. $vars = get_class_vars($realClass);
  591. $loop = isset($vars['phpQueryMethods'])
  592. && ! is_null($vars['phpQueryMethods'])
  593. ? $vars['phpQueryMethods']
  594. : get_class_methods($realClass);
  595. foreach($loop as $method) {
  596. if (! is_callable(array($realClass, $method)))
  597. continue;
  598. if (isset(self::$pluginsMethods[$method])) {
  599. throw new Exception("Duplicate method '{$method}' from plugin '{$c}' conflicts with same method from plugin '".self::$pluginsMethods[$method]."'");
  600. continue;
  601. }
  602. self::$pluginsMethods[$method] = $class;
  603. }
  604. }
  605. return true;
  606. }
  607. /**
  608. * Unloades all or specified document from memory.
  609. *
  610. * @param mixed $documentID @see phpQuery::getDocumentID() for supported types.
  611. */
  612. public static function unloadDocuments($id = null) {
  613. if (isset($id)) {
  614. if ($id = self::getDocumentID($id))
  615. unset(phpQuery::$documents[$id]);
  616. } else {
  617. foreach(phpQuery::$documents as $k => $v) {
  618. unset(phpQuery::$documents[$k]);
  619. }
  620. }
  621. }
  622. /**
  623. * Parses phpQuery object or HTML result against PHP tags and makes them active.
  624. *
  625. * @param phpQuery|string $content
  626. * @deprecated
  627. * @return string
  628. */
  629. public static function unsafePHPTags($content) {
  630. return self::markupToPHP($content);
  631. }
  632. public static function DOMNodeListToArray($DOMNodeList) {
  633. $array = array();
  634. if (! $DOMNodeList)
  635. return $array;
  636. foreach($DOMNodeList as $node)
  637. $array[] = $node;
  638. return $array;
  639. }
  640. /**
  641. * Checks if $input is HTML string, which has to start with '<'.
  642. *
  643. * @deprecated
  644. * @param String $input
  645. * @return Bool
  646. * @todo still used ?
  647. */
  648. public static function isMarkup($input) {
  649. return ! is_array($input) && substr(trim($input), 0, 1) == '<';
  650. }
  651. public static function debug($text) {
  652. if (self::$debug)
  653. print var_dump($text);
  654. }
  655. /**
  656. * Make an AJAX request.
  657. *
  658. * @param array See $options http://docs.jquery.com/Ajax/jQuery.ajax#toptions
  659. * Additional options are:
  660. * 'document' - document for global events, @see phpQuery::getDocumentID()
  661. * 'referer' - implemented
  662. * 'requested_with' - TODO; not implemented (X-Requested-With)
  663. * @return Zend_Http_Client
  664. * @link http://docs.jquery.com/Ajax/jQuery.ajax
  665. *
  666. * @TODO $options['cache']
  667. * @TODO $options['processData']
  668. * @TODO $options['xhr']
  669. * @TODO $options['data'] as string
  670. * @TODO XHR interface
  671. */
  672. public static function ajax($options = array(), $xhr = null) {
  673. $options = array_merge(
  674. self::$ajaxSettings, $options
  675. );
  676. $documentID = isset($options['document'])
  677. ? self::getDocumentID($options['document'])
  678. : null;
  679. if ($xhr) {
  680. // reuse existing XHR object, but clean it up
  681. $client = $xhr;
  682. // $client->setParameterPost(null);
  683. // $client->setParameterGet(null);
  684. $client->setAuth(false);
  685. $client->setHeaders("If-Modified-Since", null);
  686. $client->setHeaders("Referer", null);
  687. $client->resetParameters();
  688. } else {
  689. // create new XHR object
  690. require_once('Zend/Http/Client.php');
  691. $client = new Zend_Http_Client();
  692. $client->setCookieJar();
  693. }
  694. if (isset($options['timeout']))
  695. $client->setConfig(array(
  696. 'timeout' => $options['timeout'],
  697. ));
  698. // 'maxredirects' => 0,
  699. foreach(self::$ajaxAllowedHosts as $k => $host)
  700. if ($host == '.' && isset($_SERVER['HTTP_HOST']))
  701. self::$ajaxAllowedHosts[$k] = $_SERVER['HTTP_HOST'];
  702. $host = parse_url($options['url'], PHP_URL_HOST);
  703. if (! in_array($host, self::$ajaxAllowedHosts)) {
  704. throw new Exception("Request not permitted, host '$host' not present in "
  705. ."phpQuery::\$ajaxAllowedHosts");
  706. }
  707. // JSONP
  708. $jsre = "/=\\?(&|$)/";
  709. if (isset($options['dataType']) && $options['dataType'] == 'jsonp') {
  710. $jsonpCallbackParam = $options['jsonp']
  711. ? $options['jsonp'] : 'callback';
  712. if (strtolower($options['type']) == 'get') {
  713. if (! preg_match($jsre, $options['url'])) {
  714. $sep = strpos($options['url'], '?')
  715. ? '&' : '?';
  716. $options['url'] .= "$sep$jsonpCallbackParam=?";
  717. }
  718. } else if ($options['data']) {
  719. $jsonp = false;
  720. foreach($options['data'] as $n => $v) {
  721. if ($v == '?')
  722. $jsonp = true;
  723. }
  724. if (! $jsonp) {
  725. $options['data'][$jsonpCallbackParam] = '?';
  726. }
  727. }
  728. $options['dataType'] = 'json';
  729. }
  730. if (isset($options['dataType']) && $options['dataType'] == 'json') {
  731. $jsonpCallback = 'json_'.md5(microtime());
  732. $jsonpData = $jsonpUrl = false;
  733. if ($options['data']) {
  734. foreach($options['data'] as $n => $v) {
  735. if ($v == '?')
  736. $jsonpData = $n;
  737. }
  738. }
  739. if (preg_match($jsre, $options['url']))
  740. $jsonpUrl = true;
  741. if ($jsonpData !== false || $jsonpUrl) {
  742. // remember callback name for httpData()
  743. $options['_jsonp'] = $jsonpCallback;
  744. if ($jsonpData !== false)
  745. $options['data'][$jsonpData] = $jsonpCallback;
  746. if ($jsonpUrl)
  747. $options['url'] = preg_replace($jsre, "=$jsonpCallback\\1", $options['url']);
  748. }
  749. }
  750. $client->setUri($options['url']);
  751. $client->setMethod(strtoupper($options['type']));
  752. if (isset($options['referer']) && $options['referer'])
  753. $client->setHeaders('Referer', $options['referer']);
  754. $client->setHeaders(array(
  755. // 'content-type' => $options['contentType'],
  756. 'User-Agent' => 'Mozilla/5.0 (X11; U; Linux x86; en-US; rv:1.9.0.5) Gecko'
  757. .'/2008122010 Firefox/3.0.5',
  758. // TODO custom charset
  759. 'Accept-Charset' => 'ISO-8859-1,utf-8;q=0.7,*;q=0.7',
  760. // 'Connection' => 'keep-alive',
  761. // 'Accept' => 'text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8',
  762. 'Accept-Language' => 'en-us,en;q=0.5',
  763. ));
  764. if ($options['username'])
  765. $client->setAuth($options['username'], $options['password']);
  766. if (isset($options['ifModified']) && $options['ifModified'])
  767. $client->setHeaders("If-Modified-Since",
  768. self::$lastModified
  769. ? self::$lastModified
  770. : "Thu, 01 Jan 1970 00:00:00 GMT"
  771. );
  772. $client->setHeaders("Accept",
  773. isset($options['dataType'])
  774. && isset(self::$ajaxSettings['accepts'][ $options['dataType'] ])
  775. ? self::$ajaxSettings['accepts'][ $options['dataType'] ].", */*"
  776. : self::$ajaxSettings['accepts']['_default']
  777. );
  778. // TODO $options['processData']
  779. if ($options['data'] instanceof phpQueryObject) {
  780. $serialized = $options['data']->serializeArray($options['data']);
  781. $options['data'] = array();
  782. foreach($serialized as $r)
  783. $options['data'][ $r['name'] ] = $r['value'];
  784. }
  785. if (strtolower($options['type']) == 'get') {
  786. $client->setParameterGet($options['data']);
  787. } else if (strtolower($options['type']) == 'post') {
  788. $client->setEncType($options['contentType']);
  789. $client->setParameterPost($options['data']);
  790. }
  791. if (self::$active == 0 && $options['global'])
  792. phpQueryEvents::trigger($documentID, 'ajaxStart');
  793. self::$active++;
  794. // beforeSend callback
  795. if (isset($options['beforeSend']) && $options['beforeSend'])
  796. phpQuery::callbackRun($options['beforeSend'], array($client));
  797. // ajaxSend event
  798. if ($options['global'])
  799. phpQueryEvents::trigger($documentID, 'ajaxSend', array($client, $options));
  800. if (phpQuery::$debug) {
  801. self::debug("{$options['type']}: {$options['url']}\n");
  802. self::debug("Options: <pre>".var_export($options, true)."</pre>\n");
  803. // if ($client->getCookieJar())
  804. // self::debug("Cookies: <pre>".var_export($client->getCookieJar()->getMatchingCookies($options['url']), true)."</pre>\n");
  805. }
  806. // request
  807. $response = $client->request();
  808. if (phpQuery::$debug) {
  809. self::debug('Status: '.$response->getStatus().' / '.$response->getMessage());
  810. self::debug($client->getLastRequest());
  811. self::debug($response->getHeaders());
  812. }
  813. if ($response->isSuccessful()) {
  814. // XXX tempolary
  815. self::$lastModified = $response->getHeader('Last-Modified');
  816. $data = self::httpData($response->getBody(), $options['dataType'], $options);
  817. if (isset($options['success']) && $options['success'])
  818. phpQuery::callbackRun($options['success'], array($data, $response->getStatus(), $options));
  819. if ($options['global'])
  820. phpQueryEvents::trigger($documentID, 'ajaxSuccess', array($client, $options));
  821. } else {
  822. if (isset($options['error']) && $options['error'])
  823. phpQuery::callbackRun($options['error'], array($client, $response->getStatus(), $response->getMessage()));
  824. if ($options['global'])
  825. phpQueryEvents::trigger($documentID, 'ajaxError', array($client, /*$response->getStatus(),*/$response->getMessage(), $options));
  826. }
  827. if (isset($options['complete']) && $options['complete'])
  828. phpQuery::callbackRun($options['complete'], array($client, $response->getStatus()));
  829. if ($options['global'])
  830. phpQueryEvents::trigger($documentID, 'ajaxComplete', array($client, $options));
  831. if ($options['global'] && ! --self::$active)
  832. phpQueryEvents::trigger($documentID, 'ajaxStop');
  833. return $client;
  834. // if (is_null($domId))
  835. // $domId = self::$defaultDocumentID ? self::$defaultDocumentID : false;
  836. // return new phpQueryAjaxResponse($response, $domId);
  837. }
  838. protected static function httpData($data, $type, $options) {
  839. if (isset($options['dataFilter']) && $options['dataFilter'])
  840. $data = self::callbackRun($options['dataFilter'], array($data, $type));
  841. if (is_string($data)) {
  842. if ($type == "json") {
  843. if (isset($options['_jsonp']) && $options['_jsonp']) {
  844. $data = preg_replace('/^\s*\w+\((.*)\)\s*$/s', '$1', $data);
  845. }
  846. $data = self::parseJSON($data);
  847. }
  848. }
  849. return $data;
  850. }
  851. /**
  852. * Enter description here...
  853. *
  854. * @param array|phpQuery $data
  855. *
  856. */
  857. public static function param($data) {
  858. return http_build_query($data, null, '&');
  859. }
  860. public static function get($url, $data = null, $callback = null, $type = null) {
  861. if (!is_array($data)) {
  862. $callback = $data;
  863. $data = null;
  864. }
  865. // TODO some array_values on this shit
  866. return phpQuery::ajax(array(
  867. 'type' => 'GET',
  868. 'url' => $url,
  869. 'data' => $data,
  870. 'success' => $callback,
  871. 'dataType' => $type,
  872. ));
  873. }
  874. public static function post($url, $data = null, $callback = null, $type = null) {
  875. if (!is_array($data)) {
  876. $callback = $data;
  877. $data = null;
  878. }
  879. return phpQuery::ajax(array(
  880. 'type' => 'POST',
  881. 'url' => $url,
  882. 'data' => $data,
  883. 'success' => $callback,
  884. 'dataType' => $type,
  885. ));
  886. }
  887. public static function getJSON($url, $data = null, $callback = null) {
  888. if (!is_array($data)) {
  889. $callback = $data;
  890. $data = null;
  891. }
  892. // TODO some array_values on this shit
  893. return phpQuery::ajax(array(
  894. 'type' => 'GET',
  895. 'url' => $url,
  896. 'data' => $data,
  897. 'success' => $callback,
  898. 'dataType' => 'json',
  899. ));
  900. }
  901. public static function ajaxSetup($options) {
  902. self::$ajaxSettings = array_merge(
  903. self::$ajaxSettings,
  904. $options
  905. );
  906. }
  907. public static function ajaxAllowHost($host1, $host2 = null, $host3 = null) {
  908. $loop = is_array($host1)
  909. ? $host1
  910. : func_get_args();
  911. foreach($loop as $host) {
  912. if ($host && ! in_array($host, phpQuery::$ajaxAllowedHosts)) {
  913. phpQuery::$ajaxAllowedHosts[] = $host;
  914. }
  915. }
  916. }
  917. public static function ajaxAllowURL($url1, $url2 = null, $url3 = null) {
  918. $loop = is_array($url1)
  919. ? $url1
  920. : func_get_args();
  921. foreach($loop as $url)
  922. phpQuery::ajaxAllowHost(parse_url($url, PHP_URL_HOST));
  923. }
  924. /**
  925. * Returns JSON representation of $data.
  926. *
  927. * @static
  928. * @param mixed $data
  929. * @return string
  930. */
  931. public static function toJSON($data) {
  932. if (function_exists('json_encode'))
  933. return json_encode($data);
  934. require_once('Zend/Json/Encoder.php');
  935. return Zend_Json_Encoder::encode($data);
  936. }
  937. /**
  938. * Parses JSON into proper PHP type.
  939. *
  940. * @static
  941. * @param string $json
  942. * @return mixed
  943. */
  944. public static function parseJSON($json) {
  945. if (function_exists('json_decode')) {
  946. $return = json_decode(trim($json), true);
  947. // json_decode and UTF8 issues
  948. if (isset($return))
  949. return $return;
  950. }
  951. require_once('Zend/Json/Decoder.php');
  952. return Zend_Json_Decoder::decode($json);
  953. }
  954. /**
  955. * Returns source's document ID.
  956. *
  957. * @param $source DOMNode|phpQueryObject
  958. * @return string
  959. */
  960. public static function getDocumentID($source) {
  961. if ($source instanceof DOMDOCUMENT) {
  962. foreach(phpQuery::$documents as $id => $document) {
  963. if ($source->isSameNode($document->document))
  964. return $id;
  965. }
  966. } else if ($source instanceof DOMNODE) {
  967. foreach(phpQuery::$documents as $id => $document) {
  968. if ($source->ownerDocument->isSameNode($document->document))
  969. return $id;
  970. }
  971. } else if ($source instanceof phpQueryObject)
  972. return $source->getDocumentID();
  973. else if (is_string($source) && isset(phpQuery::$documents[$source]))
  974. return $source;
  975. }
  976. /**
  977. * Get DOMDocument object related to $source.
  978. * Returns null if such document doesn't exist.
  979. *
  980. * @param $source DOMNode|phpQueryObject|string
  981. * @return string
  982. */
  983. public static function getDOMDocument($source) {
  984. if ($source instanceof DOMDOCUMENT)
  985. return $source;
  986. $source = self::getDocumentID($source);
  987. return $source
  988. ? self::$documents[$id]['document']
  989. : null;
  990. }
  991. // UTILITIES
  992. // http://docs.jquery.com/Utilities
  993. /**
  994. *
  995. * @return unknown_type
  996. * @link http://docs.jquery.com/Utilities/jQuery.makeArray
  997. */
  998. public static function makeArray($obj) {
  999. $array = array();
  1000. if (is_object($object) && $object instanceof DOMNODELIST) {
  1001. foreach($object as $value)
  1002. $array[] = $value;
  1003. } else if (is_object($object) && ! ($object instanceof Iterator)) {
  1004. foreach(get_object_vars($object) as $name => $value)
  1005. $array[0][$name] = $value;
  1006. } else {
  1007. foreach($object as $name => $value)
  1008. $array[0][$name] = $value;
  1009. }
  1010. return $array;
  1011. }
  1012. public static function inArray($value, $array) {
  1013. return in_array($value, $array);
  1014. }
  1015. /**
  1016. *
  1017. * @param $object
  1018. * @param $callback
  1019. * @return unknown_type
  1020. * @link http://docs.jquery.com/Utilities/jQuery.each
  1021. */
  1022. public static function each($object, $callback, $param1 = null, $param2 = null, $param3 = null) {
  1023. $paramStructure = null;
  1024. if (func_num_args() > 2) {
  1025. $paramStructure = func_get_args();
  1026. $paramStructure = array_slice($paramStructure, 2);
  1027. }
  1028. if (is_object($object) && ! ($object instanceof Iterator)) {
  1029. foreach(get_object_vars($object) as $name => $value)
  1030. phpQuery::callbackRun($callback, array($name, $value), $paramStructure);
  1031. } else {
  1032. foreach($object as $name => $value)
  1033. phpQuery::callbackRun($callback, array($name, $value), $paramStructure);
  1034. }
  1035. }
  1036. /**
  1037. *
  1038. * @link http://docs.jquery.com/Utilities/jQuery.map
  1039. */
  1040. public static function map($array, $callback, $param1 = null, $param2 = null, $param3 = null) {
  1041. $result = array();
  1042. $paramStructure = null;
  1043. if (func_num_args() > 2) {
  1044. $paramStructure = func_get_args();
  1045. $paramStructure = array_slice($paramStructure, 2);
  1046. }
  1047. foreach($array as $v) {
  1048. $vv = phpQuery::callbackRun($callback, array($v), $paramStructure);
  1049. // $callbackArgs = $args;
  1050. // foreach($args as $i => $arg) {
  1051. // $callbackArgs[$i] = $arg instanceof CallbackParam
  1052. // ? $v
  1053. // : $arg;
  1054. // }
  1055. // $vv = call_user_func_array($callback, $callbackArgs);
  1056. if (is_array($vv)) {
  1057. foreach($vv as $vvv)
  1058. $result[] = $vvv;
  1059. } else if ($vv !== null) {
  1060. $result[] = $vv;
  1061. }
  1062. }
  1063. return $result;
  1064. }
  1065. /**
  1066. *
  1067. * @param $callback Callback
  1068. * @param $params
  1069. * @param $paramStructure
  1070. * @return unknown_type
  1071. */
  1072. public static function callbackRun($callback, $params = array(), $paramStructure = null) {
  1073. if (! $callback)
  1074. return;
  1075. if ($callback instanceof CallbackParameterToReference) {
  1076. // TODO support ParamStructure to select which $param push to reference
  1077. if (isset($params[0]))
  1078. $callback->callback = $params[0];
  1079. return true;
  1080. }
  1081. if ($callback instanceof Callback) {
  1082. $paramStructure = $callback->params;
  1083. $callback = $callback->callback;
  1084. }
  1085. if (! $paramStructure)
  1086. return call_user_func_array($callback, $params);
  1087. $p = 0;
  1088. foreach($paramStructure as $i => $v) {
  1089. $paramStructure[$i] = $v instanceof CallbackParam
  1090. ? $params[$p++]
  1091. : $v;
  1092. }
  1093. return call_user_func_array($callback, $paramStructure);
  1094. }
  1095. /**
  1096. * Merge 2 phpQuery objects.
  1097. * @param array $one
  1098. * @param array $two
  1099. * @protected
  1100. * @todo node lists, phpQueryObject
  1101. */
  1102. public static function merge($one, $two) {
  1103. $elements = $one->elements;
  1104. foreach($two->elements as $node) {
  1105. $exists = false;
  1106. foreach($elements as $node2) {
  1107. if ($node2->isSameNode($node))
  1108. $exists = true;
  1109. }
  1110. if (! $exists)
  1111. $elements[] = $node;
  1112. }
  1113. return $elements;
  1114. // $one = $one->newInstance();
  1115. // $one->elements = $elements;
  1116. // return $one;
  1117. }
  1118. /**
  1119. *
  1120. * @param $array
  1121. * @param $callback
  1122. * @param $invert
  1123. * @return unknown_type
  1124. * @link http://docs.jquery.com/Utilities/jQuery.grep
  1125. */
  1126. public static function grep($array, $callback, $invert = false) {
  1127. $result = array();
  1128. foreach($array as $k => $v) {
  1129. $r = call_user_func_array($callback, array($v, $k));
  1130. if ($r === !(bool)$invert)
  1131. $result[] = $v;
  1132. }
  1133. return $result;
  1134. }
  1135. public static function unique($array) {
  1136. return array_unique($array);
  1137. }
  1138. /**
  1139. *
  1140. * @param $function
  1141. * @return unknown_type
  1142. * @TODO there are problems with non-static methods, second parameter pass it
  1143. * but doesnt verify is method is really callable
  1144. */
  1145. public static function isFunction($function) {
  1146. return is_callable($function);
  1147. }
  1148. public static function trim($str) {
  1149. return trim($str);
  1150. }
  1151. /* PLUGINS NAMESPACE */
  1152. /**
  1153. *
  1154. * @param $url
  1155. * @param $callback
  1156. * @param $param1
  1157. * @param $param2
  1158. * @param $param3
  1159. * @return phpQueryObject
  1160. */
  1161. public static function browserGet($url, $callback, $param1 = null, $param2 = null, $param3 = null) {
  1162. if (self::plugin('WebBrowser')) {
  1163. $params = func_get_args();
  1164. return self::callbackRun(array(self::$plugins, 'browserGet'), $params);
  1165. } else {
  1166. self::debug('WebBrowser plugin not available...');
  1167. }
  1168. }
  1169. /**
  1170. *
  1171. * @param $url
  1172. * @param $data
  1173. * @param $callback
  1174. * @param $param1
  1175. * @param $param2
  1176. * @param $param3
  1177. * @return phpQueryObject
  1178. */
  1179. public static function browserPost($url, $data, $callback, $param1 = null, $param2 = null, $param3 = null) {
  1180. if (self::plugin('WebBrowser')) {
  1181. $params = func_get_args();
  1182. return self::callbackRun(array(self::$plugins, 'browserPost'), $params);
  1183. } else {
  1184. self::debug('WebBrowser plugin not available...');
  1185. }
  1186. }
  1187. /**
  1188. *
  1189. * @param $ajaxSettings
  1190. * @param $callback
  1191. * @param $param1
  1192. * @param $param2
  1193. * @param $param3
  1194. * @return phpQueryObject
  1195. */
  1196. public static function browser($ajaxSettings, $callback, $param1 = null, $param2 = null, $param3 = null) {
  1197. if (self::plugin('WebBrowser')) {
  1198. $params = func_get_args();
  1199. return self::callbackRun(array(self::$plugins, 'browser'), $params);
  1200. } else {
  1201. self::debug('WebBrowser plugin not available...');
  1202. }
  1203. }
  1204. /**
  1205. *
  1206. * @param $code
  1207. * @return string
  1208. */
  1209. public static function php($code) {
  1210. return self::code('php', $code);
  1211. }
  1212. /**
  1213. *
  1214. * @param $type
  1215. * @param $code
  1216. * @return string
  1217. */
  1218. public static function code($type, $code) {
  1219. return "<$type><!-- ".trim($code)." --></$type>";
  1220. }
  1221. public static function __callStatic($method, $params) {
  1222. return call_user_func_array(
  1223. array(phpQuery::$plugins, $method),
  1224. $params
  1225. );
  1226. }
  1227. protected static function dataSetupNode($node, $documentID) {
  1228. // search are return if alredy exists
  1229. foreach(phpQuery::$documents[$documentID]->dataNodes as $dataNode) {
  1230. if ($node->isSameNode($dataNode))
  1231. return $dataNode;
  1232. }
  1233. // if doesn't, add it
  1234. phpQuery::$documents[$documentID]->dataNodes[] = $node;
  1235. return $node;
  1236. }
  1237. protected static function dataRemoveNode($node, $documentID) {
  1238. // search are return if alredy exists
  1239. foreach(phpQuery::$documents[$documentID]->dataNodes as $k => $dataNode) {
  1240. if ($node->isSameNode($dataNode)) {
  1241. unset(self::$documents[$documentID]->dataNodes[$k]);
  1242. unset(self::$documents[$documentID]->data[ $dataNode->dataID ]);
  1243. }
  1244. }
  1245. }
  1246. public static function data($node, $name, $data, $documentID = null) {
  1247. if (! $documentID)
  1248. // TODO check if this works
  1249. $documentID = self::getDocumentID($node);
  1250. $document = phpQuery::$documents[$documentID];
  1251. $node = self::dataSetupNode($node, $documentID);
  1252. if (! isset($node->dataID))
  1253. $node->dataID = ++phpQuery::$documents[$documentID]->uuid;
  1254. $id = $node->dataID;
  1255. if (! isset($document->data[$id]))
  1256. $document->data[$id] = array();
  1257. if (! is_null($data))
  1258. $document->data[$id][$name] = $data;
  1259. if ($name) {
  1260. if (isset($document->data[$id][$name]))
  1261. return $document->data[$id][$name];
  1262. } else
  1263. return $id;
  1264. }
  1265. public static function removeData($node, $name, $documentID) {
  1266. if (! $documentID)
  1267. // TODO check if this works
  1268. $documentID = self::getDocumentID($node);
  1269. $document = phpQuery::$documents[$documentID];
  1270. $node = self::dataSetupNode($node, $documentID);
  1271. $id = $node->dataID;
  1272. if ($name) {
  1273. if (isset($document->data[$id][$name]))
  1274. unset($document->data[$id][$name]);
  1275. $name = null;
  1276. foreach($document->data[$id] as $name)
  1277. break;
  1278. if (! $name)
  1279. self::removeData($node, $name, $documentID);
  1280. } else {
  1281. self::dataRemoveNode($node, $documentID);
  1282. }
  1283. }
  1284. }
  1285. /**
  1286. * Plugins static namespace class.
  1287. *
  1288. * @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
  1289. * @package phpQuery
  1290. * @todo move plugin methods here (as statics)
  1291. */
  1292. class phpQueryPlugins {
  1293. public function __call($method, $args) {
  1294. if (isset(phpQuery::$extendStaticMethods[$method])) {
  1295. $return = call_user_func_array(
  1296. phpQuery::$extendStaticMethods[$method],
  1297. $args
  1298. );
  1299. } else if (isset(phpQuery::$pluginsStaticMethods[$method])) {
  1300. $class = phpQuery::$pluginsStaticMethods[$method];
  1301. $realClass = "phpQueryPlugin_$class";
  1302. $return = call_user_func_array(
  1303. array($realClass, $method),
  1304. $args
  1305. );
  1306. return isset($return)
  1307. ? $return
  1308. : $this;
  1309. } else
  1310. throw new Exception("Method '{$method}' doesnt exist");
  1311. }
  1312. }
  1313. /**
  1314. * Shortcut to phpQuery::pq($arg1, $context)
  1315. * Chainable.
  1316. *
  1317. * @see phpQuery::pq()
  1318. * @return phpQueryObject|QueryTemplatesSource|QueryTemplatesParse|QueryTemplatesSourceQuery
  1319. * @author Tobiasz Cudnik <tobiasz.cudnik/gmail.com>
  1320. * @package phpQuery
  1321. */
  1322. function pq($arg1, $context = null) {
  1323. $args = func_get_args();
  1324. return call_user_func_array(
  1325. array('phpQuery', 'pq'),
  1326. $args
  1327. );
  1328. }
  1329. // add plugins dir and Zend framework to include path
  1330. set_include_path(
  1331. get_include_path()
  1332. .PATH_SEPARATOR.dirname(__FILE__).'/phpQuery/'
  1333. .PATH_SEPARATOR.dirname(__FILE__).'/phpQuery/plugins/'
  1334. );
  1335. // why ? no __call nor __get for statics in php...
  1336. // XXX __callStatic will be available in PHP 5.3
  1337. phpQuery::$plugins = new phpQueryPlugins();
  1338. // include bootstrap file (personal library config)
  1339. if (file_exists(dirname(__FILE__).'/phpQuery/bootstrap.php'))
  1340. require_once dirname(__FILE__).'/phpQuery/bootstrap.php';