Iterator.cs 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224
  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. if (! _current.MoveTo (_right.Current) )
  741. _current = _right.Current.Clone ();
  742. return true;
  743. }
  744. public override XPathNavigator Current {
  745. get { return _current; }
  746. }
  747. }
  748. internal class SlashIterator : BaseIterator
  749. {
  750. private BaseIterator _iterLeft;
  751. private BaseIterator _iterRight;
  752. private NodeSet _expr;
  753. ArrayList _navStore;
  754. SortedList _iterList;
  755. bool _finished;
  756. BaseIterator _nextIterRight;
  757. public SlashIterator (BaseIterator iter, NodeSet expr, bool requireSorting) : base (iter.NamespaceManager)
  758. {
  759. _iterLeft = iter;
  760. _expr = expr;
  761. if (requireSorting)
  762. CollectResults ();
  763. }
  764. private SlashIterator (SlashIterator other) : base (other)
  765. {
  766. _iterLeft = (BaseIterator) other._iterLeft.Clone ();
  767. if (other._iterRight != null)
  768. _iterRight = (BaseIterator) other._iterRight.Clone ();
  769. _expr = other._expr;
  770. if (other._iterList != null)
  771. _iterList = (SortedList) other._iterList.Clone ();
  772. if (other._navStore != null)
  773. _navStore = (ArrayList) other._navStore.Clone ();
  774. _finished = other._finished;
  775. if (other._nextIterRight != null)
  776. _nextIterRight = (BaseIterator) other._nextIterRight.Clone ();
  777. }
  778. public override XPathNodeIterator Clone () { return new SlashIterator (this); }
  779. public override bool MoveNextCore ()
  780. {
  781. if (_finished)
  782. return false;
  783. if (_navStore != null) {
  784. // Which requires sorting::
  785. if (_navStore.Count < CurrentPosition + 1) {
  786. _finished = true;
  787. return false;
  788. }
  789. while (_navStore.Count > CurrentPosition + 1) {
  790. if (((XPathNavigator) _navStore [CurrentPosition + 1]).IsSamePosition (
  791. (XPathNavigator) _navStore [CurrentPosition]))
  792. _navStore.RemoveAt (CurrentPosition + 1);
  793. else
  794. break;
  795. }
  796. return true;
  797. }
  798. // Which does not require sorting::
  799. if (_iterRight == null) { // First time
  800. if (!_iterLeft.MoveNext ())
  801. return false;
  802. _iterRight = _expr.EvaluateNodeSet (_iterLeft);
  803. _iterList = new SortedList (XPathIteratorComparer.Instance);
  804. }
  805. while (true) {
  806. while (!_iterRight.MoveNext ()) {
  807. if (_iterList.Count > 0) {
  808. int last = _iterList.Count - 1;
  809. _iterRight = (BaseIterator) _iterList.GetByIndex (last);
  810. _iterList.RemoveAt (last);
  811. break;
  812. } else if (_nextIterRight != null) {
  813. _iterRight = _nextIterRight;
  814. _nextIterRight = null;
  815. break;
  816. } else if (!_iterLeft.MoveNext ()) {
  817. _finished = true;
  818. return false;
  819. }
  820. else
  821. _iterRight = _expr.EvaluateNodeSet (_iterLeft);
  822. }
  823. bool loop = true;
  824. while (loop) {
  825. loop = false;
  826. if (_nextIterRight == null) {
  827. bool noMoreNext = false;
  828. while (_nextIterRight == null || !_nextIterRight.MoveNext ()) {
  829. if(_iterLeft.MoveNext ())
  830. _nextIterRight = _expr.EvaluateNodeSet (_iterLeft);
  831. else {
  832. noMoreNext = true;
  833. break;
  834. }
  835. }
  836. if (noMoreNext)
  837. _nextIterRight = null; // FIXME: More efficient code. Maybe making noMoreNext class scope would be better.
  838. }
  839. if (_nextIterRight != null) {
  840. switch (_iterRight.Current.ComparePosition (_nextIterRight.Current)) {
  841. case XmlNodeOrder.After:
  842. _iterList [_iterRight] = _iterRight;
  843. _iterRight = _nextIterRight;
  844. _nextIterRight = null;
  845. loop = true;
  846. break;
  847. case XmlNodeOrder.Same:
  848. if (!_nextIterRight.MoveNext ())
  849. _nextIterRight = null;
  850. else {
  851. int last = _iterList.Count;
  852. _iterList [_nextIterRight] = _nextIterRight;
  853. if (last != _iterList.Count) {
  854. _nextIterRight = (BaseIterator) _iterList.GetByIndex (last);
  855. _iterList.RemoveAt (last);
  856. }
  857. }
  858. loop = true;
  859. break;
  860. }
  861. }
  862. }
  863. return true;
  864. }
  865. }
  866. private void CollectResults ()
  867. {
  868. _navStore = new ArrayList ();
  869. while (true) {
  870. while (_iterRight == null || !_iterRight.MoveNext ()) {
  871. if (!_iterLeft.MoveNext ()) {
  872. _navStore.Sort (XPathNavigatorComparer.Instance);
  873. return;
  874. }
  875. _iterRight = _expr.EvaluateNodeSet (_iterLeft);
  876. }
  877. XPathNavigator nav = _iterRight.Current;
  878. _navStore.Add (nav.Clone ());
  879. }
  880. }
  881. public override XPathNavigator Current {
  882. get {
  883. if (CurrentPosition <= 0) return null;
  884. if (_navStore != null) {
  885. return (XPathNavigator) _navStore [CurrentPosition - 1];
  886. } else {
  887. return _iterRight.Current;
  888. }
  889. }
  890. }
  891. public override int Count { get { return _navStore == null ? base.Count : _navStore.Count; } }
  892. }
  893. internal class PredicateIterator : BaseIterator
  894. {
  895. private BaseIterator _iter;
  896. private Expression _pred;
  897. private XPathResultType resType;
  898. private bool finished;
  899. public PredicateIterator (BaseIterator iter, Expression pred) : base (iter.NamespaceManager)
  900. {
  901. _iter = iter;
  902. _pred = pred;
  903. resType = pred.GetReturnType (iter);
  904. }
  905. private PredicateIterator (PredicateIterator other) : base (other)
  906. {
  907. _iter = (BaseIterator) other._iter.Clone ();
  908. _pred = other._pred;
  909. resType = other.resType;
  910. finished = other.finished;
  911. }
  912. public override XPathNodeIterator Clone () { return new PredicateIterator (this); }
  913. public override bool MoveNextCore ()
  914. {
  915. if (finished)
  916. return false;
  917. while (_iter.MoveNext ())
  918. {
  919. switch (resType) {
  920. case XPathResultType.Number:
  921. if (_pred.EvaluateNumber (_iter) != _iter.ComparablePosition)
  922. continue;
  923. finished = true;
  924. break;
  925. case XPathResultType.Any: {
  926. object result = _pred.Evaluate (_iter);
  927. if (result is double)
  928. {
  929. if ((double) result != _iter.ComparablePosition)
  930. continue;
  931. finished = true;
  932. }
  933. else if (!XPathFunctions.ToBoolean (result))
  934. continue;
  935. }
  936. break;
  937. default:
  938. if (!_pred.EvaluateBoolean (_iter))
  939. continue;
  940. break;
  941. }
  942. return true;
  943. }
  944. return false;
  945. }
  946. public override XPathNavigator Current { get { return _iter.Current; }}
  947. public override bool ReverseAxis {
  948. get { return _iter.ReverseAxis; }
  949. }
  950. }
  951. internal class ListIterator : BaseIterator
  952. {
  953. private IList _list;
  954. public ListIterator (BaseIterator iter, IList list) : base (iter.NamespaceManager)
  955. {
  956. if (!(list is ICloneable))
  957. throw new ArgumentException ("Target enumerator must be cloneable.");
  958. _list = list;
  959. }
  960. public ListIterator (IList list, NSResolver nsm) : base (nsm)
  961. {
  962. if (!(list is ICloneable))
  963. throw new ArgumentException ("Target enumerator must be cloneable.");
  964. _list = list;
  965. }
  966. private ListIterator (ListIterator other) : base (other)
  967. {
  968. ICloneable listClone = other._list as ICloneable;
  969. _list = (IList) listClone.Clone ();
  970. }
  971. public override XPathNodeIterator Clone () { return new ListIterator (this); }
  972. public override bool MoveNextCore ()
  973. {
  974. if (CurrentPosition >= _list.Count)
  975. return false;
  976. return true;
  977. }
  978. public override XPathNavigator Current {
  979. get {
  980. if (_list.Count == 0)
  981. return null;
  982. return (XPathNavigator) _list [CurrentPosition - 1];
  983. }
  984. }
  985. public override int Count { get { return _list.Count; } }
  986. }
  987. internal class UnionIterator : BaseIterator
  988. {
  989. private BaseIterator _left, _right;
  990. private bool keepLeft;
  991. private bool keepRight;
  992. XPathNavigator _current;
  993. public UnionIterator (BaseIterator iter, BaseIterator left, BaseIterator right) : base (iter.NamespaceManager)
  994. {
  995. _left = left;
  996. _right = right;
  997. }
  998. private UnionIterator (UnionIterator other) : base (other)
  999. {
  1000. _left = (BaseIterator) other._left.Clone ();
  1001. _right = (BaseIterator) other._right.Clone ();
  1002. keepLeft = other.keepLeft;
  1003. keepRight = other.keepRight;
  1004. if (other._current != null)
  1005. _current = other._current.Clone ();
  1006. }
  1007. public override XPathNodeIterator Clone () { return new UnionIterator (this); }
  1008. public override bool MoveNextCore ()
  1009. {
  1010. if (!keepLeft)
  1011. keepLeft = _left.MoveNext ();
  1012. if (!keepRight)
  1013. keepRight = _right.MoveNext ();
  1014. if (!keepLeft && !keepRight)
  1015. return false;
  1016. if (!keepRight) {
  1017. keepLeft = false;
  1018. SetCurrent (_left);
  1019. return true;
  1020. } else if (!keepLeft) {
  1021. keepRight = false;
  1022. SetCurrent (_right);
  1023. return true;
  1024. }
  1025. switch (_left.Current.ComparePosition (_right.Current)) {
  1026. case XmlNodeOrder.Same:
  1027. // consume both. i.e. don't output duplicate result.
  1028. keepLeft = keepRight = false;
  1029. SetCurrent (_right);
  1030. return true;
  1031. case XmlNodeOrder.Before:
  1032. case XmlNodeOrder.Unknown: // Maybe happen because of "document(a) | document(b)"
  1033. keepLeft = false;
  1034. SetCurrent (_left);
  1035. return true;
  1036. case XmlNodeOrder.After:
  1037. keepRight = false;
  1038. SetCurrent (_right);
  1039. return true;
  1040. default:
  1041. throw new InvalidOperationException ("Should not happen.");
  1042. }
  1043. }
  1044. private void SetCurrent (XPathNodeIterator iter)
  1045. {
  1046. if (_current == null)
  1047. _current = iter.Current.Clone ();
  1048. else
  1049. if (! _current.MoveTo (iter.Current) )
  1050. _current = iter.Current.Clone ();
  1051. }
  1052. public override XPathNavigator Current
  1053. {
  1054. get { return _current; }
  1055. }
  1056. }
  1057. internal class OrderedIterator : BaseIterator
  1058. {
  1059. BaseIterator iter;
  1060. ArrayList list;
  1061. int index = -1;
  1062. public OrderedIterator (BaseIterator iter)
  1063. : base (iter.NamespaceManager)
  1064. {
  1065. // if (iter.Ordered)
  1066. if (false)
  1067. this.iter = iter;
  1068. else {
  1069. list = new ArrayList ();
  1070. while (iter.MoveNext ())
  1071. list.Add (iter.Current);
  1072. list.Sort (XPathNavigatorComparer.Instance);
  1073. }
  1074. }
  1075. private OrderedIterator (OrderedIterator other, bool dummy)
  1076. : base (other)
  1077. {
  1078. if (other.iter != null)
  1079. iter = (BaseIterator) other.iter.Clone ();
  1080. list = other.list;
  1081. index = other.index;
  1082. }
  1083. public override XPathNodeIterator Clone ()
  1084. {
  1085. return new OrderedIterator (this);
  1086. }
  1087. public override bool MoveNextCore ()
  1088. {
  1089. if (iter != null)
  1090. return iter.MoveNext ();
  1091. else if (index++ < list.Count)
  1092. return true;
  1093. index--;
  1094. return false;
  1095. }
  1096. public override XPathNavigator Current {
  1097. get { return iter != null ? iter.Current : index < 0 ? null : (XPathNavigator) list [index]; }
  1098. }
  1099. }
  1100. }