rubyiterators.swg 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932
  1. /* -----------------------------------------------------------------------------
  2. * rubyiterators.swg
  3. *
  4. * Implement a C++ 'output' iterator for Ruby.
  5. *
  6. * Users can derive form the Iterator to implemet their
  7. * own iterators. As an example (real one since we use it for STL/STD
  8. * containers), the template Iterator_T does the
  9. * implementation for generic C++ iterators.
  10. * ----------------------------------------------------------------------------- */
  11. %include <std_common.i>
  12. %fragment("ConstIterator","header",fragment="<stddef.h>",fragment="GC_VALUE_definition") {
  13. namespace swig {
  14. struct stop_iteration {
  15. };
  16. /**
  17. * Abstract base class used to represent all iterators of STL containers.
  18. */
  19. struct ConstIterator {
  20. public:
  21. typedef ConstIterator self_type;
  22. protected:
  23. GC_VALUE _seq;
  24. protected:
  25. ConstIterator(VALUE seq) : _seq(seq)
  26. {
  27. }
  28. // Random access iterator methods, but not required in Ruby
  29. virtual ptrdiff_t distance(const ConstIterator &x) const
  30. {
  31. throw std::invalid_argument("distance not supported");
  32. }
  33. virtual bool equal (const ConstIterator &x) const
  34. {
  35. throw std::invalid_argument("equal not supported");
  36. }
  37. virtual self_type* advance(ptrdiff_t n)
  38. {
  39. throw std::invalid_argument("advance not supported");
  40. }
  41. public:
  42. virtual ~ConstIterator() {}
  43. // Access iterator method, required by Ruby
  44. virtual VALUE value() const {
  45. throw std::invalid_argument("value not supported");
  46. return Qnil;
  47. };
  48. virtual VALUE setValue( const VALUE& v ) {
  49. throw std::invalid_argument("value= not supported");
  50. return Qnil;
  51. }
  52. virtual self_type* next( size_t n = 1 )
  53. {
  54. return this->advance( n );
  55. }
  56. virtual self_type* previous( size_t n = 1 )
  57. {
  58. ptrdiff_t nn = n;
  59. return this->advance( -nn );
  60. }
  61. virtual VALUE to_s() const {
  62. throw std::invalid_argument("to_s not supported");
  63. return Qnil;
  64. }
  65. virtual VALUE inspect() const {
  66. throw std::invalid_argument("inspect not supported");
  67. return Qnil;
  68. }
  69. virtual ConstIterator *dup() const
  70. {
  71. throw std::invalid_argument("dup not supported");
  72. return NULL;
  73. }
  74. //
  75. // C++ common/needed methods. We emulate a bidirectional
  76. // operator, to be compatible with all the STL.
  77. // The iterator traits will then tell the STL what type of
  78. // iterator we really are.
  79. //
  80. ConstIterator() : _seq( Qnil )
  81. {
  82. }
  83. ConstIterator( const self_type& b ) : _seq( b._seq )
  84. {
  85. }
  86. self_type& operator=( const self_type& b )
  87. {
  88. _seq = b._seq;
  89. return *this;
  90. }
  91. bool operator == (const ConstIterator& x) const
  92. {
  93. return equal(x);
  94. }
  95. bool operator != (const ConstIterator& x) const
  96. {
  97. return ! operator==(x);
  98. }
  99. // Pre-decrement operator
  100. self_type& operator--()
  101. {
  102. return *previous();
  103. }
  104. // Pre-increment operator
  105. self_type& operator++()
  106. {
  107. return *next();
  108. }
  109. // Post-decrement operator
  110. self_type operator--(int)
  111. {
  112. self_type r = *this;
  113. previous();
  114. return r;
  115. }
  116. // Post-increment operator
  117. self_type operator++(int)
  118. {
  119. self_type r = *this;
  120. next();
  121. return r;
  122. }
  123. ConstIterator& operator += (ptrdiff_t n)
  124. {
  125. return *advance(n);
  126. }
  127. ConstIterator& operator -= (ptrdiff_t n)
  128. {
  129. return *advance(-n);
  130. }
  131. ConstIterator* operator + (ptrdiff_t n) const
  132. {
  133. return dup()->advance(n);
  134. }
  135. ConstIterator* operator - (ptrdiff_t n) const
  136. {
  137. return dup()->advance(-n);
  138. }
  139. ptrdiff_t operator - (const ConstIterator& x) const
  140. {
  141. return x.distance(*this);
  142. }
  143. static swig_type_info* descriptor() {
  144. static int init = 0;
  145. static swig_type_info* desc = 0;
  146. if (!init) {
  147. desc = SWIG_TypeQuery("swig::ConstIterator *");
  148. init = 1;
  149. }
  150. return desc;
  151. }
  152. };
  153. /**
  154. * Abstract base class used to represent all non-const iterators of STL containers.
  155. *
  156. */
  157. struct Iterator : public ConstIterator {
  158. public:
  159. typedef Iterator self_type;
  160. protected:
  161. Iterator(VALUE seq) : ConstIterator(seq)
  162. {
  163. }
  164. virtual self_type* advance(ptrdiff_t n)
  165. {
  166. throw std::invalid_argument("operation not supported");
  167. }
  168. public:
  169. static swig_type_info* descriptor() {
  170. static int init = 0;
  171. static swig_type_info* desc = 0;
  172. if (!init) {
  173. desc = SWIG_TypeQuery("swig::Iterator *");
  174. init = 1;
  175. }
  176. return desc;
  177. }
  178. virtual Iterator *dup() const
  179. {
  180. throw std::invalid_argument("dup not supported");
  181. return NULL;
  182. }
  183. virtual self_type* next( size_t n = 1 )
  184. {
  185. return this->advance( n );
  186. }
  187. virtual self_type* previous( size_t n = 1 )
  188. {
  189. ptrdiff_t nn = n;
  190. return this->advance( -nn );
  191. }
  192. bool operator == (const ConstIterator& x) const
  193. {
  194. return equal(x);
  195. }
  196. bool operator != (const Iterator& x) const
  197. {
  198. return ! operator==(x);
  199. }
  200. Iterator& operator += (ptrdiff_t n)
  201. {
  202. return *advance(n);
  203. }
  204. Iterator& operator -= (ptrdiff_t n)
  205. {
  206. return *advance(-n);
  207. }
  208. Iterator* operator + (ptrdiff_t n) const
  209. {
  210. return dup()->advance(n);
  211. }
  212. Iterator* operator - (ptrdiff_t n) const
  213. {
  214. return dup()->advance(-n);
  215. }
  216. ptrdiff_t operator - (const Iterator& x) const
  217. {
  218. return x.distance(*this);
  219. }
  220. };
  221. }
  222. }
  223. %fragment("ConstIterator_T","header",fragment="<stddef.h>",fragment="ConstIterator",fragment="StdTraits",fragment="StdIteratorTraits") {
  224. namespace swig {
  225. /**
  226. * Templated base classes for all custom const_iterators.
  227. *
  228. */
  229. template<typename OutConstIterator>
  230. class ConstIterator_T : public ConstIterator
  231. {
  232. public:
  233. typedef OutConstIterator const_iter;
  234. typedef typename std::iterator_traits<const_iter>::value_type value_type;
  235. typedef ConstIterator_T<const_iter> self_type;
  236. protected:
  237. virtual bool equal (const ConstIterator &iter) const
  238. {
  239. const self_type *iters = dynamic_cast<const self_type *>(&iter);
  240. if (iters) {
  241. return (current == iters->get_current());
  242. } else {
  243. throw std::invalid_argument("bad iterator type");
  244. }
  245. }
  246. virtual ptrdiff_t distance(const ConstIterator &iter) const
  247. {
  248. const self_type *iters = dynamic_cast<const self_type *>(&iter);
  249. if (iters) {
  250. return std::distance(current, iters->get_current());
  251. } else {
  252. throw std::invalid_argument("bad iterator type");
  253. }
  254. }
  255. virtual ConstIterator* advance(ptrdiff_t n)
  256. {
  257. std::advance( current, n );
  258. return this;
  259. }
  260. public:
  261. ConstIterator_T() : ConstIterator(Qnil)
  262. {
  263. }
  264. ConstIterator_T(const_iter curr, VALUE seq = Qnil)
  265. : ConstIterator(seq), current(curr)
  266. {
  267. }
  268. const const_iter& get_current() const
  269. {
  270. return current;
  271. }
  272. const value_type& operator*() const
  273. {
  274. return *current;
  275. }
  276. virtual VALUE inspect() const
  277. {
  278. VALUE ret = rb_str_new2("#<");
  279. ret = rb_str_cat2( ret, rb_obj_classname(_seq) );
  280. ret = rb_str_cat2( ret, "::const_iterator " );
  281. VALUE cur = value();
  282. ret = rb_str_concat( ret, rb_inspect(cur) );
  283. ret = rb_str_cat2( ret, ">" );
  284. return ret;
  285. }
  286. virtual VALUE to_s() const
  287. {
  288. VALUE ret = rb_str_new2( rb_obj_classname(_seq) );
  289. ret = rb_str_cat2( ret, "::const_iterator " );
  290. VALUE cur = value();
  291. ret = rb_str_concat( ret, rb_obj_as_string(cur) );
  292. return ret;
  293. }
  294. protected:
  295. const_iter current;
  296. };
  297. /**
  298. * Templated base classes for all custom non-const iterators.
  299. *
  300. */
  301. template<typename InOutIterator>
  302. class Iterator_T : public Iterator
  303. {
  304. public:
  305. typedef InOutIterator nonconst_iter;
  306. // Make this class iterator STL compatible, by using iterator_traits
  307. typedef typename std::iterator_traits<nonconst_iter >::iterator_category iterator_category;
  308. typedef typename std::iterator_traits<nonconst_iter >::value_type value_type;
  309. typedef typename std::iterator_traits<nonconst_iter >::difference_type difference_type;
  310. typedef typename std::iterator_traits<nonconst_iter >::pointer pointer;
  311. typedef typename std::iterator_traits<nonconst_iter >::reference reference;
  312. typedef Iterator base;
  313. typedef Iterator_T< nonconst_iter > self_type;
  314. protected:
  315. virtual bool equal (const ConstIterator &iter) const
  316. {
  317. const self_type *iters = dynamic_cast<const self_type *>(&iter);
  318. if (iters) {
  319. return (current == iters->get_current());
  320. } else {
  321. throw std::invalid_argument("bad iterator type");
  322. }
  323. }
  324. virtual ptrdiff_t distance(const ConstIterator &iter) const
  325. {
  326. const self_type *iters = dynamic_cast<const self_type *>(&iter);
  327. if (iters) {
  328. return std::distance(current, iters->get_current());
  329. } else {
  330. throw std::invalid_argument("bad iterator type");
  331. }
  332. }
  333. virtual Iterator* advance(ptrdiff_t n)
  334. {
  335. std::advance( current, n );
  336. return this;
  337. }
  338. public:
  339. Iterator_T(nonconst_iter curr, VALUE seq = Qnil)
  340. : Iterator(seq), current(curr)
  341. {
  342. }
  343. const nonconst_iter& get_current() const
  344. {
  345. return current;
  346. }
  347. self_type& operator=( const self_type& b )
  348. {
  349. base::operator=( b );
  350. return *this;
  351. }
  352. self_type& operator=( const value_type& b )
  353. {
  354. *current = b;
  355. return *this;
  356. }
  357. const value_type& operator*() const
  358. {
  359. return *current;
  360. }
  361. value_type& operator*()
  362. {
  363. return *current;
  364. }
  365. virtual VALUE inspect() const
  366. {
  367. VALUE ret = rb_str_new2("#<");
  368. ret = rb_str_cat2( ret, rb_obj_classname(_seq) );
  369. ret = rb_str_cat2( ret, "::iterator " );
  370. VALUE cur = value();
  371. ret = rb_str_concat( ret, rb_inspect(cur) );
  372. ret = rb_str_cat2( ret, ">" );
  373. return ret;
  374. }
  375. virtual VALUE to_s() const
  376. {
  377. VALUE ret = rb_str_new2( rb_obj_classname(_seq) );
  378. ret = rb_str_cat2( ret, "::iterator " );
  379. VALUE cur = value();
  380. ret = rb_str_concat( ret, rb_obj_as_string(cur) );
  381. return ret;
  382. }
  383. protected:
  384. nonconst_iter current;
  385. };
  386. /**
  387. * Auxiliary functor to store the value of a ruby object inside
  388. * a reference of a compatible C++ type. ie: Ruby -> C++
  389. *
  390. */
  391. template <class ValueType>
  392. struct asval_oper
  393. {
  394. typedef ValueType value_type;
  395. typedef bool result_type;
  396. bool operator()(VALUE obj, value_type& v) const
  397. {
  398. return ( swig::asval< value_type >(obj, &v) == SWIG_OK );
  399. }
  400. };
  401. /**
  402. * Auxiliary functor to return a ruby object from a C++ type.
  403. * ie: C++ -> Ruby
  404. *
  405. */
  406. template <class ValueType>
  407. struct from_oper
  408. {
  409. typedef const ValueType& argument_type;
  410. typedef VALUE result_type;
  411. result_type operator()(argument_type v) const
  412. {
  413. return swig::from(v);
  414. }
  415. };
  416. /**
  417. * ConstIterator class for a const_iterator with no end() boundaries.
  418. *
  419. */
  420. template<typename OutConstIterator,
  421. typename ValueType = typename std::iterator_traits<OutConstIterator>::value_type,
  422. typename FromOper = from_oper<ValueType> >
  423. class ConstIteratorOpen_T : public ConstIterator_T<OutConstIterator>
  424. {
  425. public:
  426. FromOper from;
  427. typedef OutConstIterator const_iter;
  428. typedef ValueType value_type;
  429. typedef ConstIterator_T<const_iter> base;
  430. typedef ConstIteratorOpen_T<OutConstIterator, ValueType, FromOper> self_type;
  431. ConstIteratorOpen_T(const_iter curr, VALUE seq = Qnil)
  432. : ConstIterator_T<OutConstIterator>(curr, seq)
  433. {
  434. }
  435. virtual VALUE value() const {
  436. return from(static_cast<const value_type&>(*(base::current)));
  437. }
  438. ConstIterator *dup() const
  439. {
  440. return new self_type(*this);
  441. }
  442. };
  443. /**
  444. * Iterator class for an iterator with no end() boundaries.
  445. *
  446. */
  447. template<typename InOutIterator,
  448. typename ValueType = typename std::iterator_traits<InOutIterator>::value_type,
  449. typename FromOper = from_oper<ValueType>,
  450. typename AsvalOper = asval_oper<ValueType> >
  451. class IteratorOpen_T : public Iterator_T<InOutIterator>
  452. {
  453. public:
  454. FromOper from;
  455. AsvalOper asval;
  456. typedef InOutIterator nonconst_iter;
  457. typedef ValueType value_type;
  458. typedef Iterator_T<nonconst_iter> base;
  459. typedef IteratorOpen_T<InOutIterator, ValueType, FromOper, AsvalOper> self_type;
  460. public:
  461. IteratorOpen_T(nonconst_iter curr, VALUE seq = Qnil)
  462. : Iterator_T<InOutIterator>(curr, seq)
  463. {
  464. }
  465. virtual VALUE value() const {
  466. return from(static_cast<const value_type&>(*(base::current)));
  467. }
  468. virtual VALUE setValue( const VALUE& v )
  469. {
  470. value_type& dst = *base::current;
  471. if ( asval(v, dst) ) return v;
  472. return Qnil;
  473. }
  474. Iterator *dup() const
  475. {
  476. return new self_type(*this);
  477. }
  478. };
  479. /**
  480. * ConstIterator class for a const_iterator where begin() and end() boundaries are known.
  481. *
  482. */
  483. template<typename OutConstIterator,
  484. typename ValueType = typename std::iterator_traits<OutConstIterator>::value_type,
  485. typename FromOper = from_oper<ValueType> >
  486. class ConstIteratorClosed_T : public ConstIterator_T<OutConstIterator>
  487. {
  488. public:
  489. FromOper from;
  490. typedef OutConstIterator const_iter;
  491. typedef ValueType value_type;
  492. typedef ConstIterator_T<const_iter> base;
  493. typedef ConstIteratorClosed_T<OutConstIterator, ValueType, FromOper> self_type;
  494. protected:
  495. virtual ConstIterator* advance(ptrdiff_t n)
  496. {
  497. std::advance( base::current, n );
  498. if ( base::current == end )
  499. throw stop_iteration();
  500. return this;
  501. }
  502. public:
  503. ConstIteratorClosed_T(const_iter curr, const_iter first,
  504. const_iter last, VALUE seq = Qnil)
  505. : ConstIterator_T<OutConstIterator>(curr, seq), begin(first), end(last)
  506. {
  507. }
  508. virtual VALUE value() const {
  509. if (base::current == end) {
  510. throw stop_iteration();
  511. } else {
  512. return from(static_cast<const value_type&>(*(base::current)));
  513. }
  514. }
  515. ConstIterator *dup() const
  516. {
  517. return new self_type(*this);
  518. }
  519. private:
  520. const_iter begin;
  521. const_iter end;
  522. };
  523. /**
  524. * Iterator class for a iterator where begin() and end() boundaries are known.
  525. *
  526. */
  527. template<typename InOutIterator,
  528. typename ValueType = typename std::iterator_traits<InOutIterator>::value_type,
  529. typename FromOper = from_oper<ValueType>,
  530. typename AsvalOper = asval_oper<ValueType> >
  531. class IteratorClosed_T : public Iterator_T<InOutIterator>
  532. {
  533. public:
  534. FromOper from;
  535. AsvalOper asval;
  536. typedef InOutIterator nonconst_iter;
  537. typedef ValueType value_type;
  538. typedef Iterator_T<nonconst_iter> base;
  539. typedef IteratorClosed_T<InOutIterator, ValueType, FromOper, AsvalOper> self_type;
  540. protected:
  541. virtual Iterator* advance(ptrdiff_t n)
  542. {
  543. std::advance( base::current, n );
  544. if ( base::current == end )
  545. throw stop_iteration();
  546. return this;
  547. }
  548. public:
  549. IteratorClosed_T(nonconst_iter curr, nonconst_iter first,
  550. nonconst_iter last, VALUE seq = Qnil)
  551. : Iterator_T<InOutIterator>(curr, seq), begin(first), end(last)
  552. {
  553. }
  554. virtual VALUE value() const {
  555. if (base::current == end) {
  556. throw stop_iteration();
  557. } else {
  558. return from(static_cast<const value_type&>(*(base::current)));
  559. }
  560. }
  561. // Iterator setter method, required by Ruby
  562. virtual VALUE setValue( const VALUE& v )
  563. {
  564. if (base::current == end)
  565. throw stop_iteration();
  566. value_type& dst = *base::current;
  567. if ( asval( v, dst ) ) return v;
  568. return Qnil;
  569. }
  570. Iterator *dup() const
  571. {
  572. return new self_type(*this);
  573. }
  574. private:
  575. nonconst_iter begin;
  576. nonconst_iter end;
  577. };
  578. /* Partial specialization for bools which don't allow de-referencing */
  579. template< typename InOutIterator, typename FromOper, typename AsvalOper >
  580. class IteratorOpen_T< InOutIterator, bool, FromOper, AsvalOper > :
  581. public Iterator_T<InOutIterator>
  582. {
  583. public:
  584. FromOper from;
  585. AsvalOper asval;
  586. typedef InOutIterator nonconst_iter;
  587. typedef bool value_type;
  588. typedef Iterator_T<nonconst_iter> base;
  589. typedef IteratorOpen_T<InOutIterator, bool, FromOper, AsvalOper> self_type;
  590. IteratorOpen_T(nonconst_iter curr, VALUE seq = Qnil)
  591. : Iterator_T<InOutIterator>(curr, seq)
  592. {
  593. }
  594. virtual VALUE value() const {
  595. return from(static_cast<const value_type&>(*(base::current)));
  596. }
  597. virtual VALUE setValue( const VALUE& v )
  598. {
  599. bool tmp = *base::current;
  600. if ( asval( v, tmp ) )
  601. {
  602. *base::current = tmp;
  603. return v;
  604. }
  605. return Qnil;
  606. }
  607. Iterator *dup() const
  608. {
  609. return new self_type(*this);
  610. }
  611. };
  612. /* Partial specialization for bools which don't allow de-referencing */
  613. template< typename InOutIterator, typename FromOper, typename AsvalOper >
  614. class IteratorClosed_T< InOutIterator, bool, FromOper, AsvalOper > :
  615. public Iterator_T<InOutIterator>
  616. {
  617. public:
  618. FromOper from;
  619. AsvalOper asval;
  620. typedef InOutIterator nonconst_iter;
  621. typedef bool value_type;
  622. typedef Iterator_T<nonconst_iter> base;
  623. typedef IteratorClosed_T<InOutIterator, bool, FromOper, AsvalOper> self_type;
  624. protected:
  625. virtual Iterator* advance(ptrdiff_t n)
  626. {
  627. std::advance( base::current, n );
  628. if ( base::current == end )
  629. throw stop_iteration();
  630. return this;
  631. }
  632. public:
  633. IteratorClosed_T(nonconst_iter curr, nonconst_iter first,
  634. nonconst_iter last, VALUE seq = Qnil)
  635. : Iterator_T<InOutIterator>(curr, seq), begin(first), end(last)
  636. {
  637. }
  638. virtual VALUE value() const {
  639. if (base::current == end) {
  640. throw stop_iteration();
  641. } else {
  642. return from(static_cast<const value_type&>(*(base::current)));
  643. }
  644. }
  645. virtual VALUE setValue( const VALUE& v )
  646. {
  647. if (base::current == end)
  648. throw stop_iteration();
  649. bool tmp = *base::current;
  650. if ( asval( v, tmp ) )
  651. {
  652. *base::current = tmp;
  653. return v;
  654. }
  655. return Qnil;
  656. }
  657. Iterator *dup() const
  658. {
  659. return new self_type(*this);
  660. }
  661. private:
  662. nonconst_iter begin;
  663. nonconst_iter end;
  664. };
  665. /**
  666. * Helper function used to wrap a bounded const_iterator. This is to be used in
  667. * a %typemap(out), for example.
  668. *
  669. */
  670. template<typename InOutIter>
  671. inline Iterator*
  672. make_nonconst_iterator(const InOutIter& current, const InOutIter& begin,
  673. const InOutIter& end, VALUE seq = Qnil)
  674. {
  675. return new IteratorClosed_T<InOutIter>(current, begin, end, seq);
  676. }
  677. /**
  678. * Helper function used to wrap an unbounded const_iterator. This is to be used in
  679. * a %typemap(out), for example.
  680. *
  681. */
  682. template<typename InOutIter>
  683. inline Iterator*
  684. make_nonconst_iterator(const InOutIter& current, VALUE seq = Qnil)
  685. {
  686. return new IteratorOpen_T<InOutIter>(current, seq);
  687. }
  688. /**
  689. * Helper function used to wrap a bounded const_iterator. This is to be used in
  690. * a %typemap(out), for example.
  691. *
  692. */
  693. template<typename OutIter>
  694. inline ConstIterator*
  695. make_const_iterator(const OutIter& current, const OutIter& begin,
  696. const OutIter& end, VALUE seq = Qnil)
  697. {
  698. return new ConstIteratorClosed_T<OutIter>(current, begin, end, seq);
  699. }
  700. /**
  701. * Helper function used to wrap an unbounded const_iterator. This is to be used in
  702. * a %typemap(out), for example.
  703. *
  704. */
  705. template<typename OutIter>
  706. inline ConstIterator*
  707. make_const_iterator(const OutIter& current, VALUE seq = Qnil)
  708. {
  709. return new ConstIteratorOpen_T<OutIter>(current, seq);
  710. }
  711. }
  712. }
  713. %fragment("ConstIterator");
  714. //
  715. // This part is just so SWIG is aware of the base abstract iterator class.
  716. //
  717. namespace swig
  718. {
  719. /*
  720. Throw a StopIteration exception
  721. */
  722. %ignore stop_iteration;
  723. struct stop_iteration {};
  724. %typemap(throws) stop_iteration {
  725. (void)$1;
  726. SWIG_Ruby_ExceptionType(NULL, Qnil);
  727. SWIG_fail;
  728. }
  729. /*
  730. Mark methods that return new objects
  731. */
  732. %newobject ConstIterator::dup;
  733. %newobject ConstIterator::operator + (ptrdiff_t n) const;
  734. %newobject ConstIterator::operator - (ptrdiff_t n) const;
  735. %nodirector ConstIterator;
  736. %catches(swig::stop_iteration) ConstIterator::value() const;
  737. %catches(swig::stop_iteration) ConstIterator::incr(size_t n = 1);
  738. %catches(swig::stop_iteration) ConstIterator::decr(size_t n = 1);
  739. %catches(std::invalid_argument) ConstIterator::distance(const ConstIterator &x) const;
  740. %catches(std::invalid_argument) ConstIterator::equal (const ConstIterator &x) const;
  741. %catches(swig::stop_iteration) ConstIterator::next();
  742. %catches(swig::stop_iteration) ConstIterator::previous();
  743. %catches(swig::stop_iteration) ConstIterator::advance(ptrdiff_t n);
  744. %catches(swig::stop_iteration) ConstIterator::operator += (ptrdiff_t n);
  745. %catches(swig::stop_iteration) ConstIterator::operator -= (ptrdiff_t n);
  746. %catches(swig::stop_iteration) ConstIterator::operator + (ptrdiff_t n) const;
  747. %catches(swig::stop_iteration) ConstIterator::operator - (ptrdiff_t n) const;
  748. struct ConstIterator
  749. {
  750. protected:
  751. ConstIterator(VALUE seq);
  752. public:
  753. virtual ~ConstIterator();
  754. // Access iterator method, required by Ruby
  755. virtual VALUE value() const;
  756. // C++ common/needed methods
  757. virtual ConstIterator *dup() const;
  758. virtual VALUE inspect() const;
  759. virtual VALUE to_s() const;
  760. virtual ConstIterator* next(size_t n = 1);
  761. virtual ConstIterator* previous(size_t n = 1);
  762. bool operator == (const ConstIterator& x) const;
  763. ConstIterator* operator + (ptrdiff_t n) const;
  764. ConstIterator* operator - (ptrdiff_t n) const;
  765. ptrdiff_t operator - (const ConstIterator& x) const;
  766. };
  767. struct Iterator : public ConstIterator
  768. {
  769. %rename("value=") setValue( const VALUE& v );
  770. virtual VALUE setValue( const VALUE& v );
  771. virtual Iterator *dup() const;
  772. virtual Iterator* next(size_t n = 1);
  773. virtual Iterator* previous(size_t n = 1);
  774. virtual VALUE inspect() const;
  775. virtual VALUE to_s() const;
  776. bool operator == (const Iterator& x) const;
  777. Iterator* operator + (ptrdiff_t n) const;
  778. Iterator* operator - (ptrdiff_t n) const;
  779. ptrdiff_t operator - (const Iterator& x) const;
  780. };
  781. }