Iterator.cs 29 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222
  1. //
  2. // System.Xml.XPath.BaseIterator
  3. //
  4. // Author:
  5. // Piers Haken ([email protected])
  6. // Atsushi Enomoto ([email protected])
  7. //
  8. // (C) 2002 Piers Haken
  9. // (C) 2003 Atsushi Enomoto
  10. //
  11. //
  12. // Permission is hereby granted, free of charge, to any person obtaining
  13. // a copy of this software and associated documentation files (the
  14. // "Software"), to deal in the Software without restriction, including
  15. // without limitation the rights to use, copy, modify, merge, publish,
  16. // distribute, sublicense, and/or sell copies of the Software, and to
  17. // permit persons to whom the Software is furnished to do so, subject to
  18. // the following conditions:
  19. //
  20. // The above copyright notice and this permission notice shall be
  21. // included in all copies or substantial portions of the Software.
  22. //
  23. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  24. // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
  25. // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  26. // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
  27. // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
  28. // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
  29. // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
  30. //
  31. using System;
  32. using System.Collections;
  33. using System.Xml;
  34. using System.Xml.XPath;
  35. using System.Xml.Xsl;
  36. #if NET_2_0
  37. using NSResolver = System.Xml.IXmlNamespaceResolver;
  38. #else
  39. using NSResolver = System.Xml.XmlNamespaceManager;
  40. #endif
  41. namespace System.Xml.XPath
  42. {
  43. internal abstract class BaseIterator : XPathNodeIterator
  44. {
  45. NSResolver _nsm;
  46. int position;
  47. internal BaseIterator (BaseIterator other)
  48. {
  49. _nsm = other._nsm;
  50. position = other.position;
  51. }
  52. internal BaseIterator (NSResolver nsm)
  53. {
  54. _nsm = nsm;
  55. }
  56. public NSResolver NamespaceManager
  57. {
  58. get { return _nsm; }
  59. set { _nsm = value; }
  60. }
  61. public virtual bool ReverseAxis {
  62. get { return false; }
  63. }
  64. public int ComparablePosition {
  65. get {
  66. if (ReverseAxis) {
  67. int diff = Count - CurrentPosition + 1;
  68. return diff < 1 ? 1 : diff;
  69. }
  70. else
  71. return CurrentPosition;
  72. }
  73. }
  74. public override int CurrentPosition {
  75. get { return position; }
  76. }
  77. public override bool MoveNext ()
  78. {
  79. if (!MoveNextCore ())
  80. return false;
  81. position++;
  82. if (Current != null &&
  83. Current.NodeType == XPathNodeType.Whitespace &&
  84. _nsm is XsltContext &&
  85. !((XsltContext) _nsm).PreserveWhitespace (Current))
  86. return MoveNext ();
  87. return true;
  88. }
  89. public abstract bool MoveNextCore ();
  90. public override string ToString ()
  91. {
  92. if (Current != null)
  93. return Current.NodeType.ToString () + "[" + CurrentPosition + "] : " + Current.Name + " = " + Current.Value;
  94. else
  95. return this.GetType().ToString () + "[" + CurrentPosition + "]";
  96. }
  97. }
  98. internal class WrapperIterator : BaseIterator
  99. {
  100. XPathNodeIterator iter;
  101. public WrapperIterator (XPathNodeIterator iter, NSResolver nsm)
  102. : base (nsm)
  103. {
  104. this.iter = iter;
  105. }
  106. private WrapperIterator (WrapperIterator other)
  107. : base (other)
  108. {
  109. iter = other.iter.Clone ();
  110. }
  111. public override XPathNodeIterator Clone ()
  112. {
  113. return new WrapperIterator (this);
  114. }
  115. public override bool MoveNextCore ()
  116. {
  117. return iter.MoveNext ();
  118. }
  119. public override XPathNavigator Current {
  120. get { return iter.Current; }
  121. }
  122. }
  123. internal abstract class SimpleIterator : BaseIterator
  124. {
  125. protected readonly XPathNavigator _nav;
  126. protected XPathNavigator _current;
  127. public SimpleIterator (BaseIterator iter) : base (iter.NamespaceManager)
  128. {
  129. _nav = iter.Current.Clone ();
  130. }
  131. protected SimpleIterator (SimpleIterator other, bool clone) : base (other)
  132. {
  133. _nav = other._nav.Clone ();
  134. }
  135. public SimpleIterator (XPathNavigator nav, NSResolver nsm) : base (nsm)
  136. {
  137. _nav = nav.Clone ();
  138. }
  139. public override XPathNavigator Current {
  140. get {
  141. if (_current == null) // position == 0
  142. _current = _nav.Clone ();
  143. return _current;
  144. }
  145. }
  146. }
  147. internal class SelfIterator : SimpleIterator
  148. {
  149. public SelfIterator (BaseIterator iter) : base (iter) {}
  150. public SelfIterator (XPathNavigator nav, NSResolver nsm) : base (nav, nsm) {}
  151. protected SelfIterator (SelfIterator other, bool clone) : base (other, true)
  152. {
  153. }
  154. public override XPathNodeIterator Clone () { return new SelfIterator (this, true); }
  155. public override bool MoveNextCore ()
  156. {
  157. if (CurrentPosition == 0)
  158. {
  159. return true;
  160. }
  161. return false;
  162. }
  163. public override XPathNavigator Current {
  164. get { return _nav; }
  165. }
  166. }
  167. internal class NullIterator : SelfIterator
  168. {
  169. public NullIterator (BaseIterator iter) : base (iter) {}
  170. public NullIterator (XPathNavigator nav) : this (nav, null) {}
  171. public NullIterator (XPathNavigator nav, NSResolver nsm) : base (nav, nsm) {}
  172. private NullIterator (NullIterator other) : base (other, true) {}
  173. public override XPathNodeIterator Clone () { return new NullIterator (this); }
  174. public override bool MoveNextCore ()
  175. {
  176. return false;
  177. }
  178. }
  179. internal class ParensIterator : BaseIterator
  180. {
  181. BaseIterator _iter;
  182. public ParensIterator (BaseIterator iter) : base (iter.NamespaceManager)
  183. {
  184. _iter = iter;
  185. }
  186. private ParensIterator (ParensIterator other) : base (other)
  187. {
  188. _iter = (BaseIterator) other._iter.Clone ();
  189. }
  190. public override XPathNodeIterator Clone () { return new ParensIterator (this); }
  191. public override bool MoveNextCore ()
  192. {
  193. return _iter.MoveNext ();
  194. }
  195. public override XPathNavigator Current { get { return _iter.Current; }}
  196. public override int Count { get { return _iter.Count; } }
  197. }
  198. internal class ParentIterator : SimpleIterator
  199. {
  200. bool canMove;
  201. public ParentIterator (BaseIterator iter) : base (iter)
  202. {
  203. canMove = _nav.MoveToParent ();
  204. _current = _nav;
  205. }
  206. private ParentIterator (ParentIterator other, bool dummy) : base (other, true)
  207. {
  208. _current = _nav;
  209. canMove = other.canMove;
  210. }
  211. public ParentIterator (XPathNavigator nav, NSResolver nsm) : base (nav, nsm) {}
  212. public override XPathNodeIterator Clone () { return new ParentIterator (this, true); }
  213. public override bool MoveNextCore ()
  214. {
  215. if (!canMove)
  216. return false;
  217. canMove = false;
  218. return true;
  219. }
  220. }
  221. internal class ChildIterator : SimpleIterator
  222. {
  223. public ChildIterator (BaseIterator iter) : base (iter) {}
  224. private ChildIterator (ChildIterator other) : base (other, true) {}
  225. public override XPathNodeIterator Clone () { return new ChildIterator (this); }
  226. public override bool MoveNextCore ()
  227. {
  228. bool fSuccess = (CurrentPosition == 0) ? _nav.MoveToFirstChild () : _nav.MoveToNext ();
  229. if (fSuccess) {
  230. Current.MoveTo (_nav);
  231. }
  232. return fSuccess;
  233. }
  234. }
  235. internal class FollowingSiblingIterator : SimpleIterator
  236. {
  237. public FollowingSiblingIterator (BaseIterator iter) : base (iter) {}
  238. private FollowingSiblingIterator (FollowingSiblingIterator other) : base (other, true) {}
  239. public override XPathNodeIterator Clone () { return new FollowingSiblingIterator (this); }
  240. public override bool MoveNextCore ()
  241. {
  242. switch (_nav.NodeType) {
  243. case XPathNodeType.Attribute:
  244. case XPathNodeType.Namespace:
  245. // They have no siblings.
  246. return false;
  247. }
  248. if (_nav.MoveToNext ())
  249. {
  250. Current.MoveTo (_nav);
  251. return true;
  252. }
  253. return false;
  254. }
  255. }
  256. internal class PrecedingSiblingIterator : SimpleIterator
  257. {
  258. bool finished;
  259. bool started;
  260. XPathNavigator startPosition;
  261. public PrecedingSiblingIterator (BaseIterator iter) : base (iter)
  262. {
  263. startPosition = iter.Current.Clone ();
  264. }
  265. private PrecedingSiblingIterator (PrecedingSiblingIterator other) : base (other, true)
  266. {
  267. startPosition = other.startPosition;
  268. started = other.started;
  269. finished = other.finished;
  270. }
  271. public override XPathNodeIterator Clone () { return new PrecedingSiblingIterator (this); }
  272. public override bool MoveNextCore ()
  273. {
  274. if (finished)
  275. return false;
  276. if (!started) {
  277. started = true;
  278. switch (_nav.NodeType) {
  279. case XPathNodeType.Attribute:
  280. case XPathNodeType.Namespace:
  281. // They have no siblings.
  282. finished = true;
  283. return false;
  284. }
  285. _nav.MoveToFirst ();
  286. if (!_nav.IsSamePosition (startPosition)) {
  287. Current.MoveTo (_nav);
  288. return true;
  289. }
  290. } else {
  291. if (!_nav.MoveToNext ()) {
  292. finished = true;
  293. return false;
  294. }
  295. }
  296. if (_nav.ComparePosition (startPosition) != XmlNodeOrder.Before) {
  297. // Note that if _nav contains only 1 node, it won't be Same.
  298. finished = true;
  299. return false;
  300. } else {
  301. Current.MoveTo (_nav);
  302. return true;
  303. }
  304. }
  305. public override bool ReverseAxis {
  306. get { return true; }
  307. }
  308. }
  309. internal class AncestorIterator : SimpleIterator
  310. {
  311. int currentPosition;
  312. ArrayList navigators;
  313. XPathNavigator startPosition;
  314. public AncestorIterator (BaseIterator iter) : base (iter)
  315. {
  316. startPosition = iter.Current.Clone ();
  317. }
  318. private AncestorIterator (AncestorIterator other)
  319. : base (other, true)
  320. {
  321. startPosition = other.startPosition;
  322. if (other.navigators != null)
  323. navigators = (ArrayList) other.navigators;
  324. currentPosition = other.currentPosition;
  325. }
  326. public override XPathNodeIterator Clone ()
  327. {
  328. return new AncestorIterator (this);
  329. }
  330. private void CollectResults ()
  331. {
  332. navigators = new ArrayList ();
  333. XPathNavigator ancestors = startPosition.Clone ();
  334. if (!ancestors.MoveToParent ())
  335. return;
  336. while (ancestors.NodeType != XPathNodeType.Root) {
  337. navigators.Add (ancestors.Clone ());
  338. ancestors.MoveToParent ();
  339. }
  340. currentPosition = navigators.Count;
  341. }
  342. public override bool MoveNextCore ()
  343. {
  344. if (navigators == null) {
  345. CollectResults ();
  346. if (startPosition.NodeType != XPathNodeType.Root) {
  347. // First time it returns Root
  348. _nav.MoveToRoot ();
  349. Current.MoveTo (_nav);
  350. return true;
  351. }
  352. }
  353. if (currentPosition == 0)
  354. return false;
  355. _nav.MoveTo ((XPathNavigator) navigators [--currentPosition]);
  356. Current.MoveTo (_nav);
  357. return true;
  358. }
  359. public override bool ReverseAxis {
  360. get { return true; }
  361. }
  362. public override int Count {
  363. get {
  364. if (navigators == null)
  365. CollectResults ();
  366. return navigators.Count;
  367. }
  368. }
  369. }
  370. internal class AncestorOrSelfIterator : SimpleIterator
  371. {
  372. int currentPosition;
  373. ArrayList navigators;
  374. XPathNavigator startPosition;
  375. public AncestorOrSelfIterator (BaseIterator iter) : base (iter)
  376. {
  377. startPosition = iter.Current.Clone ();
  378. }
  379. private AncestorOrSelfIterator (AncestorOrSelfIterator other)
  380. : base (other, true)
  381. {
  382. startPosition = other.startPosition;
  383. if (other.navigators != null)
  384. navigators = (ArrayList) other.navigators.Clone ();
  385. currentPosition = other.currentPosition;
  386. }
  387. public override XPathNodeIterator Clone ()
  388. {
  389. return new AncestorOrSelfIterator (this);
  390. }
  391. private void CollectResults ()
  392. {
  393. navigators = new ArrayList ();
  394. XPathNavigator ancestors = startPosition.Clone ();
  395. if (!ancestors.MoveToParent ())
  396. return;
  397. while (ancestors.NodeType != XPathNodeType.Root) {
  398. navigators.Add (ancestors.Clone ());
  399. ancestors.MoveToParent ();
  400. }
  401. currentPosition = navigators.Count;
  402. }
  403. public override bool MoveNextCore ()
  404. {
  405. if (navigators == null) {
  406. CollectResults ();
  407. if (startPosition.NodeType != XPathNodeType.Root) {
  408. // First time it returns Root
  409. _nav.MoveToRoot ();
  410. Current.MoveTo (_nav);
  411. return true;
  412. }
  413. }
  414. if (currentPosition == -1)
  415. return false;
  416. if (currentPosition-- == 0) {
  417. _nav.MoveTo (startPosition);
  418. Current.MoveTo (_nav);
  419. return true; // returns self.
  420. }
  421. _nav.MoveTo ((XPathNavigator) navigators [currentPosition]);
  422. Current.MoveTo (_nav);
  423. return true;
  424. }
  425. public override bool ReverseAxis {
  426. get { return true; }
  427. }
  428. public override int Count {
  429. get {
  430. if (navigators == null)
  431. CollectResults ();
  432. return navigators.Count + 1;
  433. }
  434. }
  435. }
  436. internal class DescendantIterator : SimpleIterator
  437. {
  438. private int _depth;
  439. private bool _finished;
  440. public DescendantIterator (BaseIterator iter) : base (iter) {}
  441. private DescendantIterator (DescendantIterator other) : base (other)
  442. {
  443. _depth = other._depth;
  444. _finished = other._finished;
  445. }
  446. public override XPathNodeIterator Clone () { return new DescendantIterator (this); }
  447. public override bool MoveNextCore ()
  448. {
  449. if (_finished)
  450. return false;
  451. if (_nav.MoveToFirstChild ())
  452. {
  453. _depth ++;
  454. Current.MoveTo (_nav);
  455. return true;
  456. }
  457. while (_depth != 0)
  458. {
  459. if (_nav.MoveToNext ())
  460. {
  461. Current.MoveTo (_nav);
  462. return true;
  463. }
  464. if (!_nav.MoveToParent ()) // should NEVER fail!
  465. throw new XPathException ("There seems some bugs on the XPathNavigator implementation class.");
  466. _depth --;
  467. }
  468. _finished = true;
  469. return false;
  470. }
  471. }
  472. internal class DescendantOrSelfIterator : SimpleIterator
  473. {
  474. private int _depth;
  475. private bool _finished;
  476. public DescendantOrSelfIterator (BaseIterator iter) : base (iter) {}
  477. private DescendantOrSelfIterator (DescendantOrSelfIterator other) : base (other, true)
  478. {
  479. _depth = other._depth;
  480. }
  481. public override XPathNodeIterator Clone () { return new DescendantOrSelfIterator (this); }
  482. public override bool MoveNextCore ()
  483. {
  484. if (_finished)
  485. return false;
  486. if (CurrentPosition == 0)
  487. {
  488. // self
  489. Current.MoveTo (_nav);
  490. return true;
  491. }
  492. if (_nav.MoveToFirstChild ())
  493. {
  494. _depth ++;
  495. Current.MoveTo (_nav);
  496. return true;
  497. }
  498. while (_depth != 0)
  499. {
  500. if (_nav.MoveToNext ())
  501. {
  502. Current.MoveTo (_nav);
  503. return true;
  504. }
  505. if (!_nav.MoveToParent ()) // should NEVER fail!
  506. throw new XPathException ("There seems some bugs on the XPathNavigator implementation class.");
  507. _depth --;
  508. }
  509. _finished = true;
  510. return false;
  511. }
  512. }
  513. internal class FollowingIterator : SimpleIterator
  514. {
  515. private bool _finished = false;
  516. public FollowingIterator (BaseIterator iter) : base (iter) {}
  517. private FollowingIterator (FollowingIterator other) : base (other, true) {}
  518. public override XPathNodeIterator Clone () { return new FollowingIterator (this); }
  519. public override bool MoveNextCore ()
  520. {
  521. if (_finished)
  522. return false;
  523. bool checkChildren = true;
  524. if (CurrentPosition == 0)
  525. {
  526. checkChildren = false;
  527. switch (_nav.NodeType) {
  528. case XPathNodeType.Attribute:
  529. case XPathNodeType.Namespace:
  530. _nav.MoveToParent ();
  531. checkChildren = true;
  532. break;
  533. default:
  534. if (_nav.MoveToNext ())
  535. {
  536. Current.MoveTo (_nav);
  537. return true;
  538. } else {
  539. while (_nav.MoveToParent ()) {
  540. if (_nav.MoveToNext ()) {
  541. Current.MoveTo (_nav);
  542. return true;
  543. }
  544. }
  545. }
  546. break;
  547. }
  548. }
  549. if (checkChildren)
  550. {
  551. if (_nav.MoveToFirstChild ())
  552. {
  553. Current.MoveTo (_nav);
  554. return true;
  555. }
  556. do
  557. {
  558. if (_nav.MoveToNext ())
  559. {
  560. Current.MoveTo (_nav);
  561. return true;
  562. }
  563. }
  564. while (_nav.MoveToParent ());
  565. }
  566. _finished = true;
  567. return false;
  568. }
  569. }
  570. internal class PrecedingIterator : SimpleIterator
  571. {
  572. bool finished;
  573. bool started;
  574. XPathNavigator startPosition;
  575. public PrecedingIterator (BaseIterator iter) : base (iter)
  576. {
  577. startPosition = iter.Current.Clone ();
  578. }
  579. private PrecedingIterator (PrecedingIterator other) : base (other, true)
  580. {
  581. startPosition = other.startPosition;
  582. started = other.started;
  583. finished = other.finished;
  584. }
  585. public override XPathNodeIterator Clone () { return new PrecedingIterator (this); }
  586. public override bool MoveNextCore ()
  587. {
  588. if (finished)
  589. return false;
  590. if (!started) {
  591. started = true;
  592. _nav.MoveToRoot ();
  593. }
  594. bool loop = true;
  595. while (loop) {
  596. while (!_nav.MoveToFirstChild ()) {
  597. while (!_nav.MoveToNext ()) {
  598. if (!_nav.MoveToParent ()) { // Should not finish, at least before startPosition.
  599. finished = true;
  600. return false;
  601. }
  602. }
  603. break;
  604. }
  605. if (_nav.IsDescendant (startPosition))
  606. continue;
  607. loop = false;
  608. break;
  609. }
  610. if (_nav.ComparePosition (startPosition) != XmlNodeOrder.Before) {
  611. // Note that if _nav contains only 1 node, it won't be Same.
  612. finished = true;
  613. return false;
  614. } else {
  615. Current.MoveTo (_nav);
  616. return true;
  617. }
  618. }
  619. public override bool ReverseAxis {
  620. get { return true; }
  621. }
  622. }
  623. internal class NamespaceIterator : SimpleIterator
  624. {
  625. public NamespaceIterator (BaseIterator iter) : base (iter) {}
  626. private NamespaceIterator (NamespaceIterator other) : base (other, true) {}
  627. public override XPathNodeIterator Clone () { return new NamespaceIterator (this); }
  628. public override bool MoveNextCore ()
  629. {
  630. if (CurrentPosition == 0)
  631. {
  632. if (_nav.MoveToFirstNamespace ())
  633. {
  634. Current.MoveTo (_nav);
  635. return true;
  636. }
  637. }
  638. else if (_nav.MoveToNextNamespace ())
  639. {
  640. Current.MoveTo (_nav);
  641. return true;
  642. }
  643. return false;
  644. }
  645. }
  646. internal class AttributeIterator : SimpleIterator
  647. {
  648. public AttributeIterator (BaseIterator iter) : base (iter) {}
  649. private AttributeIterator (AttributeIterator other) : base (other, true) {}
  650. public override XPathNodeIterator Clone () { return new AttributeIterator (this); }
  651. public override bool MoveNextCore ()
  652. {
  653. if (CurrentPosition == 0)
  654. {
  655. if (_nav.MoveToFirstAttribute ())
  656. {
  657. Current.MoveTo (_nav);
  658. return true;
  659. }
  660. }
  661. else if (_nav.MoveToNextAttribute ())
  662. {
  663. Current.MoveTo (_nav);
  664. return true;
  665. }
  666. return false;
  667. }
  668. }
  669. internal class AxisIterator : BaseIterator
  670. {
  671. private SimpleIterator _iter;
  672. private NodeTest _test;
  673. string name, ns;
  674. XPathNodeType matchType;
  675. public AxisIterator (SimpleIterator iter, NodeTest test) : base (iter.NamespaceManager)
  676. {
  677. _iter = iter;
  678. _test = test;
  679. test.GetInfo (out name, out ns, out matchType, NamespaceManager);
  680. // if (name != null)
  681. // name = Current.NameTable.Add (name);
  682. // if (ns != null)
  683. // ns = Current.NameTable.Add (ns);
  684. }
  685. private AxisIterator (AxisIterator other) : base (other)
  686. {
  687. _iter = (SimpleIterator) other._iter.Clone ();
  688. _test = other._test;
  689. name = other.name;
  690. ns = other.ns;
  691. matchType = other.matchType;
  692. }
  693. public override XPathNodeIterator Clone () { return new AxisIterator (this); }
  694. public override bool MoveNextCore ()
  695. {
  696. while (_iter.MoveNext ())
  697. {
  698. if (_test.Match (NamespaceManager, Current))
  699. {
  700. return true;
  701. }
  702. }
  703. return false;
  704. }
  705. public override XPathNavigator Current { get { return _iter.Current; }}
  706. public override bool ReverseAxis {
  707. get { return _iter.ReverseAxis; }
  708. }
  709. }
  710. internal class SimpleSlashIterator : BaseIterator
  711. {
  712. private NodeSet _expr;
  713. private BaseIterator _left, _right;
  714. private XPathNavigator _current;
  715. public SimpleSlashIterator (BaseIterator left, NodeSet expr)
  716. : base (left.NamespaceManager)
  717. {
  718. this._left = left;
  719. this._expr = expr;
  720. }
  721. private SimpleSlashIterator (SimpleSlashIterator other)
  722. : base (other)
  723. {
  724. _expr = other._expr;
  725. _left = (BaseIterator) other._left.Clone ();
  726. if (other._right != null)
  727. _right = (BaseIterator) other._right.Clone ();
  728. }
  729. public override XPathNodeIterator Clone () { return new SimpleSlashIterator (this); }
  730. public override bool MoveNextCore ()
  731. {
  732. while (_right == null || !_right.MoveNext ()) {
  733. if (!_left.MoveNext ())
  734. return false;
  735. _right = _expr.EvaluateNodeSet (_left);
  736. }
  737. if (_current == null)
  738. _current = _right.Current.Clone ();
  739. else
  740. _current.MoveTo (_right.Current);
  741. return true;
  742. }
  743. public override XPathNavigator Current {
  744. get { return _current; }
  745. }
  746. }
  747. internal class SlashIterator : BaseIterator
  748. {
  749. private BaseIterator _iterLeft;
  750. private BaseIterator _iterRight;
  751. private NodeSet _expr;
  752. ArrayList _navStore;
  753. SortedList _iterList;
  754. bool _finished;
  755. BaseIterator _nextIterRight;
  756. public SlashIterator (BaseIterator iter, NodeSet expr, bool requireSorting) : base (iter.NamespaceManager)
  757. {
  758. _iterLeft = iter;
  759. _expr = expr;
  760. if (requireSorting)
  761. CollectResults ();
  762. }
  763. private SlashIterator (SlashIterator other) : base (other)
  764. {
  765. _iterLeft = (BaseIterator) other._iterLeft.Clone ();
  766. if (other._iterRight != null)
  767. _iterRight = (BaseIterator) other._iterRight.Clone ();
  768. _expr = other._expr;
  769. if (other._iterList != null)
  770. _iterList = (SortedList) other._iterList.Clone ();
  771. if (other._navStore != null)
  772. _navStore = (ArrayList) other._navStore.Clone ();
  773. _finished = other._finished;
  774. if (other._nextIterRight != null)
  775. _nextIterRight = (BaseIterator) other._nextIterRight.Clone ();
  776. }
  777. public override XPathNodeIterator Clone () { return new SlashIterator (this); }
  778. public override bool MoveNextCore ()
  779. {
  780. if (_finished)
  781. return false;
  782. if (_navStore != null) {
  783. // Which requires sorting::
  784. if (_navStore.Count < CurrentPosition + 1) {
  785. _finished = true;
  786. return false;
  787. }
  788. while (_navStore.Count > CurrentPosition + 1) {
  789. if (((XPathNavigator) _navStore [CurrentPosition + 1]).IsSamePosition (
  790. (XPathNavigator) _navStore [CurrentPosition]))
  791. _navStore.RemoveAt (CurrentPosition + 1);
  792. else
  793. break;
  794. }
  795. return true;
  796. }
  797. // Which does not require sorting::
  798. if (_iterRight == null) { // First time
  799. if (!_iterLeft.MoveNext ())
  800. return false;
  801. _iterRight = _expr.EvaluateNodeSet (_iterLeft);
  802. _iterList = new SortedList (XPathIteratorComparer.Instance);
  803. }
  804. while (true) {
  805. while (!_iterRight.MoveNext ()) {
  806. if (_iterList.Count > 0) {
  807. int last = _iterList.Count - 1;
  808. _iterRight = (BaseIterator) _iterList.GetByIndex (last);
  809. _iterList.RemoveAt (last);
  810. break;
  811. } else if (_nextIterRight != null) {
  812. _iterRight = _nextIterRight;
  813. _nextIterRight = null;
  814. break;
  815. } else if (!_iterLeft.MoveNext ()) {
  816. _finished = true;
  817. return false;
  818. }
  819. else
  820. _iterRight = _expr.EvaluateNodeSet (_iterLeft);
  821. }
  822. bool loop = true;
  823. while (loop) {
  824. loop = false;
  825. if (_nextIterRight == null) {
  826. bool noMoreNext = false;
  827. while (_nextIterRight == null || !_nextIterRight.MoveNext ()) {
  828. if(_iterLeft.MoveNext ())
  829. _nextIterRight = _expr.EvaluateNodeSet (_iterLeft);
  830. else {
  831. noMoreNext = true;
  832. break;
  833. }
  834. }
  835. if (noMoreNext)
  836. _nextIterRight = null; // FIXME: More efficient code. Maybe making noMoreNext class scope would be better.
  837. }
  838. if (_nextIterRight != null) {
  839. switch (_iterRight.Current.ComparePosition (_nextIterRight.Current)) {
  840. case XmlNodeOrder.After:
  841. _iterList [_iterRight] = _iterRight;
  842. _iterRight = _nextIterRight;
  843. _nextIterRight = null;
  844. loop = true;
  845. break;
  846. case XmlNodeOrder.Same:
  847. if (!_nextIterRight.MoveNext ())
  848. _nextIterRight = null;
  849. else {
  850. int last = _iterList.Count;
  851. _iterList [_nextIterRight] = _nextIterRight;
  852. if (last != _iterList.Count) {
  853. _nextIterRight = (BaseIterator) _iterList.GetByIndex (last);
  854. _iterList.RemoveAt (last);
  855. }
  856. }
  857. loop = true;
  858. break;
  859. }
  860. }
  861. }
  862. return true;
  863. }
  864. }
  865. private void CollectResults ()
  866. {
  867. _navStore = new ArrayList ();
  868. while (true) {
  869. while (_iterRight == null || !_iterRight.MoveNext ()) {
  870. if (!_iterLeft.MoveNext ()) {
  871. _navStore.Sort (XPathNavigatorComparer.Instance);
  872. return;
  873. }
  874. _iterRight = _expr.EvaluateNodeSet (_iterLeft);
  875. }
  876. XPathNavigator nav = _iterRight.Current;
  877. _navStore.Add (nav.Clone ());
  878. }
  879. }
  880. public override XPathNavigator Current {
  881. get {
  882. if (CurrentPosition <= 0) return null;
  883. if (_navStore != null) {
  884. return (XPathNavigator) _navStore [CurrentPosition - 1];
  885. } else {
  886. return _iterRight.Current;
  887. }
  888. }
  889. }
  890. public override int Count { get { return _navStore == null ? base.Count : _navStore.Count; } }
  891. }
  892. internal class PredicateIterator : BaseIterator
  893. {
  894. private BaseIterator _iter;
  895. private Expression _pred;
  896. private XPathResultType resType;
  897. private bool finished;
  898. public PredicateIterator (BaseIterator iter, Expression pred) : base (iter.NamespaceManager)
  899. {
  900. _iter = iter;
  901. _pred = pred;
  902. resType = pred.GetReturnType (iter);
  903. }
  904. private PredicateIterator (PredicateIterator other) : base (other)
  905. {
  906. _iter = (BaseIterator) other._iter.Clone ();
  907. _pred = other._pred;
  908. resType = other.resType;
  909. finished = other.finished;
  910. }
  911. public override XPathNodeIterator Clone () { return new PredicateIterator (this); }
  912. public override bool MoveNextCore ()
  913. {
  914. if (finished)
  915. return false;
  916. while (_iter.MoveNext ())
  917. {
  918. switch (resType) {
  919. case XPathResultType.Number:
  920. if (_pred.EvaluateNumber (_iter) != _iter.ComparablePosition)
  921. continue;
  922. finished = true;
  923. break;
  924. case XPathResultType.Any: {
  925. object result = _pred.Evaluate (_iter);
  926. if (result is double)
  927. {
  928. if ((double) result != _iter.ComparablePosition)
  929. continue;
  930. finished = true;
  931. }
  932. else if (!XPathFunctions.ToBoolean (result))
  933. continue;
  934. }
  935. break;
  936. default:
  937. if (!_pred.EvaluateBoolean (_iter))
  938. continue;
  939. break;
  940. }
  941. return true;
  942. }
  943. return false;
  944. }
  945. public override XPathNavigator Current { get { return _iter.Current; }}
  946. public override bool ReverseAxis {
  947. get { return _iter.ReverseAxis; }
  948. }
  949. }
  950. internal class ListIterator : BaseIterator
  951. {
  952. private IList _list;
  953. public ListIterator (BaseIterator iter, IList list) : base (iter.NamespaceManager)
  954. {
  955. if (!(list is ICloneable))
  956. throw new ArgumentException ("Target enumerator must be cloneable.");
  957. _list = list;
  958. }
  959. public ListIterator (IList list, NSResolver nsm) : base (nsm)
  960. {
  961. if (!(list is ICloneable))
  962. throw new ArgumentException ("Target enumerator must be cloneable.");
  963. _list = list;
  964. }
  965. private ListIterator (ListIterator other) : base (other)
  966. {
  967. ICloneable listClone = other._list as ICloneable;
  968. _list = (IList) listClone.Clone ();
  969. }
  970. public override XPathNodeIterator Clone () { return new ListIterator (this); }
  971. public override bool MoveNextCore ()
  972. {
  973. if (CurrentPosition >= _list.Count)
  974. return false;
  975. return true;
  976. }
  977. public override XPathNavigator Current {
  978. get {
  979. if (_list.Count == 0)
  980. return null;
  981. return (XPathNavigator) _list [CurrentPosition - 1];
  982. }
  983. }
  984. public override int Count { get { return _list.Count; } }
  985. }
  986. internal class UnionIterator : BaseIterator
  987. {
  988. private BaseIterator _left, _right;
  989. private bool keepLeft;
  990. private bool keepRight;
  991. XPathNavigator _current;
  992. public UnionIterator (BaseIterator iter, BaseIterator left, BaseIterator right) : base (iter.NamespaceManager)
  993. {
  994. _left = left;
  995. _right = right;
  996. }
  997. private UnionIterator (UnionIterator other) : base (other)
  998. {
  999. _left = (BaseIterator) other._left.Clone ();
  1000. _right = (BaseIterator) other._right.Clone ();
  1001. keepLeft = other.keepLeft;
  1002. keepRight = other.keepRight;
  1003. if (other._current != null)
  1004. _current = other._current.Clone ();
  1005. }
  1006. public override XPathNodeIterator Clone () { return new UnionIterator (this); }
  1007. public override bool MoveNextCore ()
  1008. {
  1009. if (!keepLeft)
  1010. keepLeft = _left.MoveNext ();
  1011. if (!keepRight)
  1012. keepRight = _right.MoveNext ();
  1013. if (!keepLeft && !keepRight)
  1014. return false;
  1015. if (!keepRight) {
  1016. keepLeft = false;
  1017. SetCurrent (_left);
  1018. return true;
  1019. } else if (!keepLeft) {
  1020. keepRight = false;
  1021. SetCurrent (_right);
  1022. return true;
  1023. }
  1024. switch (_left.Current.ComparePosition (_right.Current)) {
  1025. case XmlNodeOrder.Same:
  1026. // consume both. i.e. don't output duplicate result.
  1027. keepLeft = keepRight = false;
  1028. SetCurrent (_right);
  1029. return true;
  1030. case XmlNodeOrder.Before:
  1031. case XmlNodeOrder.Unknown: // Maybe happen because of "document(a) | document(b)"
  1032. keepLeft = false;
  1033. SetCurrent (_left);
  1034. return true;
  1035. case XmlNodeOrder.After:
  1036. keepRight = false;
  1037. SetCurrent (_right);
  1038. return true;
  1039. default:
  1040. throw new InvalidOperationException ("Should not happen.");
  1041. }
  1042. }
  1043. private void SetCurrent (XPathNodeIterator iter)
  1044. {
  1045. if (_current == null)
  1046. _current = iter.Current.Clone ();
  1047. else
  1048. _current.MoveTo (iter.Current);
  1049. }
  1050. public override XPathNavigator Current
  1051. {
  1052. get { return _current; }
  1053. }
  1054. }
  1055. internal class OrderedIterator : BaseIterator
  1056. {
  1057. BaseIterator iter;
  1058. ArrayList list;
  1059. int index = -1;
  1060. public OrderedIterator (BaseIterator iter)
  1061. : base (iter.NamespaceManager)
  1062. {
  1063. // if (iter.Ordered)
  1064. if (false)
  1065. this.iter = iter;
  1066. else {
  1067. list = new ArrayList ();
  1068. while (iter.MoveNext ())
  1069. list.Add (iter.Current);
  1070. list.Sort (XPathNavigatorComparer.Instance);
  1071. }
  1072. }
  1073. private OrderedIterator (OrderedIterator other, bool dummy)
  1074. : base (other)
  1075. {
  1076. if (other.iter != null)
  1077. iter = (BaseIterator) other.iter.Clone ();
  1078. list = other.list;
  1079. index = other.index;
  1080. }
  1081. public override XPathNodeIterator Clone ()
  1082. {
  1083. return new OrderedIterator (this);
  1084. }
  1085. public override bool MoveNextCore ()
  1086. {
  1087. if (iter != null)
  1088. return iter.MoveNext ();
  1089. else if (index++ < list.Count)
  1090. return true;
  1091. index--;
  1092. return false;
  1093. }
  1094. public override XPathNavigator Current {
  1095. get { return iter != null ? iter.Current : index < 0 ? null : (XPathNavigator) list [index]; }
  1096. }
  1097. }
  1098. }