Chris@16
|
1 // Boost.Range library
|
Chris@16
|
2 //
|
Chris@16
|
3 // Copyright Neil Groves 2009. Use, modification and
|
Chris@16
|
4 // distribution is subject to the Boost Software License, Version
|
Chris@16
|
5 // 1.0. (See accompanying file LICENSE_1_0.txt or copy at
|
Chris@16
|
6 // http://www.boost.org/LICENSE_1_0.txt)
|
Chris@16
|
7 //
|
Chris@16
|
8 // Acknowledgements:
|
Chris@16
|
9 // aschoedl contributed an improvement to the determination
|
Chris@16
|
10 // of the Reference type parameter.
|
Chris@16
|
11 //
|
Chris@101
|
12 // Leonid Gershanovich reported Trac ticket 7376 about the dereference operator
|
Chris@101
|
13 // requiring identical reference types due to using the ternary if.
|
Chris@101
|
14 //
|
Chris@16
|
15 // For more information, see http://www.boost.org/libs/range/
|
Chris@16
|
16 //
|
Chris@16
|
17 #ifndef BOOST_RANGE_DETAIL_JOIN_ITERATOR_HPP_INCLUDED
|
Chris@16
|
18 #define BOOST_RANGE_DETAIL_JOIN_ITERATOR_HPP_INCLUDED
|
Chris@16
|
19
|
Chris@16
|
20 #include <iterator>
|
Chris@16
|
21 #include <boost/assert.hpp>
|
Chris@16
|
22 #include <boost/iterator/iterator_traits.hpp>
|
Chris@16
|
23 #include <boost/iterator/iterator_facade.hpp>
|
Chris@16
|
24 #include <boost/range/begin.hpp>
|
Chris@16
|
25 #include <boost/range/end.hpp>
|
Chris@16
|
26 #include <boost/range/empty.hpp>
|
Chris@16
|
27 #include <boost/range/detail/demote_iterator_traversal_tag.hpp>
|
Chris@16
|
28 #include <boost/range/value_type.hpp>
|
Chris@101
|
29 #include <boost/type_traits/add_const.hpp>
|
Chris@101
|
30 #include <boost/type_traits/add_reference.hpp>
|
Chris@101
|
31 #include <boost/type_traits/remove_const.hpp>
|
Chris@101
|
32 #include <boost/type_traits/remove_reference.hpp>
|
Chris@16
|
33 #include <boost/next_prior.hpp>
|
Chris@16
|
34
|
Chris@16
|
35 namespace boost
|
Chris@16
|
36 {
|
Chris@16
|
37 namespace range_detail
|
Chris@16
|
38 {
|
Chris@16
|
39
|
Chris@16
|
40 template<typename Iterator1, typename Iterator2>
|
Chris@16
|
41 struct join_iterator_link
|
Chris@16
|
42 {
|
Chris@16
|
43 public:
|
Chris@16
|
44 join_iterator_link(Iterator1 last1, Iterator2 first2)
|
Chris@16
|
45 : last1(last1)
|
Chris@16
|
46 , first2(first2)
|
Chris@16
|
47 {
|
Chris@16
|
48 }
|
Chris@16
|
49
|
Chris@16
|
50 Iterator1 last1;
|
Chris@16
|
51 Iterator2 first2;
|
Chris@16
|
52
|
Chris@16
|
53 private:
|
Chris@16
|
54 join_iterator_link() /* = delete */ ;
|
Chris@16
|
55 };
|
Chris@16
|
56
|
Chris@16
|
57 class join_iterator_begin_tag {};
|
Chris@16
|
58 class join_iterator_end_tag {};
|
Chris@16
|
59
|
Chris@16
|
60 template<typename Iterator1
|
Chris@16
|
61 , typename Iterator2
|
Chris@16
|
62 , typename Reference
|
Chris@16
|
63 >
|
Chris@16
|
64 class join_iterator_union
|
Chris@16
|
65 {
|
Chris@16
|
66 public:
|
Chris@16
|
67 typedef Iterator1 iterator1_t;
|
Chris@16
|
68 typedef Iterator2 iterator2_t;
|
Chris@16
|
69
|
Chris@16
|
70 join_iterator_union() {}
|
Chris@16
|
71 join_iterator_union(unsigned int /*selected*/, const iterator1_t& it1, const iterator2_t& it2) : m_it1(it1), m_it2(it2) {}
|
Chris@16
|
72
|
Chris@16
|
73 iterator1_t& it1() { return m_it1; }
|
Chris@16
|
74 const iterator1_t& it1() const { return m_it1; }
|
Chris@16
|
75
|
Chris@16
|
76 iterator2_t& it2() { return m_it2; }
|
Chris@16
|
77 const iterator2_t& it2() const { return m_it2; }
|
Chris@16
|
78
|
Chris@16
|
79 Reference dereference(unsigned int selected) const
|
Chris@16
|
80 {
|
Chris@101
|
81 if (selected)
|
Chris@101
|
82 return *m_it2;
|
Chris@101
|
83 return *m_it1;
|
Chris@16
|
84 }
|
Chris@16
|
85
|
Chris@16
|
86 bool equal(const join_iterator_union& other, unsigned int selected) const
|
Chris@16
|
87 {
|
Chris@16
|
88 return selected
|
Chris@16
|
89 ? m_it2 == other.m_it2
|
Chris@16
|
90 : m_it1 == other.m_it1;
|
Chris@16
|
91 }
|
Chris@16
|
92
|
Chris@16
|
93 private:
|
Chris@16
|
94 iterator1_t m_it1;
|
Chris@16
|
95 iterator2_t m_it2;
|
Chris@16
|
96 };
|
Chris@16
|
97
|
Chris@16
|
98 template<class Iterator, class Reference>
|
Chris@16
|
99 class join_iterator_union<Iterator, Iterator, Reference>
|
Chris@16
|
100 {
|
Chris@16
|
101 public:
|
Chris@16
|
102 typedef Iterator iterator1_t;
|
Chris@16
|
103 typedef Iterator iterator2_t;
|
Chris@16
|
104
|
Chris@16
|
105 join_iterator_union() {}
|
Chris@16
|
106
|
Chris@16
|
107 join_iterator_union(unsigned int selected, const iterator1_t& it1, const iterator2_t& it2)
|
Chris@16
|
108 : m_it(selected ? it2 : it1)
|
Chris@16
|
109 {
|
Chris@16
|
110 }
|
Chris@16
|
111
|
Chris@16
|
112 iterator1_t& it1() { return m_it; }
|
Chris@16
|
113 const iterator1_t& it1() const { return m_it; }
|
Chris@16
|
114
|
Chris@16
|
115 iterator2_t& it2() { return m_it; }
|
Chris@16
|
116 const iterator2_t& it2() const { return m_it; }
|
Chris@16
|
117
|
Chris@16
|
118 Reference dereference(unsigned int) const
|
Chris@16
|
119 {
|
Chris@16
|
120 return *m_it;
|
Chris@16
|
121 }
|
Chris@16
|
122
|
Chris@101
|
123 bool equal(const join_iterator_union& other,
|
Chris@101
|
124 unsigned int /*selected*/) const
|
Chris@16
|
125 {
|
Chris@16
|
126 return m_it == other.m_it;
|
Chris@16
|
127 }
|
Chris@16
|
128
|
Chris@16
|
129 private:
|
Chris@16
|
130 iterator1_t m_it;
|
Chris@16
|
131 };
|
Chris@16
|
132
|
Chris@16
|
133 template<typename Iterator1
|
Chris@16
|
134 , typename Iterator2
|
Chris@16
|
135 , typename ValueType = typename iterator_value<Iterator1>::type
|
Chris@16
|
136 // find least demanding, commonly supported reference type, in the order &, const&, and by-value:
|
Chris@16
|
137 , typename Reference = typename mpl::if_c<
|
Chris@16
|
138 !is_reference<typename iterator_reference<Iterator1>::type>::value
|
Chris@16
|
139 || !is_reference<typename iterator_reference<Iterator2>::type>::value,
|
Chris@16
|
140 typename remove_const<
|
Chris@16
|
141 typename remove_reference<
|
Chris@16
|
142 typename iterator_reference<Iterator1>::type
|
Chris@16
|
143 >::type
|
Chris@16
|
144 >::type,
|
Chris@16
|
145 typename mpl::if_c<
|
Chris@16
|
146 is_const<
|
Chris@16
|
147 typename remove_reference<
|
Chris@16
|
148 typename iterator_reference<Iterator1>::type
|
Chris@16
|
149 >::type
|
Chris@16
|
150 >::value
|
Chris@16
|
151 || is_const<
|
Chris@16
|
152 typename remove_reference<
|
Chris@16
|
153 typename iterator_reference<Iterator2>::type
|
Chris@16
|
154 >::type
|
Chris@16
|
155 >::value,
|
Chris@16
|
156 typename add_const<
|
Chris@101
|
157 typename iterator_reference<Iterator1>::type
|
Chris@16
|
158 >::type,
|
Chris@16
|
159 typename iterator_reference<Iterator1>::type
|
Chris@16
|
160 >::type
|
Chris@16
|
161 >::type
|
Chris@16
|
162 , typename Traversal = typename demote_iterator_traversal_tag<
|
Chris@16
|
163 typename iterator_traversal<Iterator1>::type
|
Chris@16
|
164 , typename iterator_traversal<Iterator2>::type>::type
|
Chris@16
|
165 >
|
Chris@16
|
166 class join_iterator
|
Chris@16
|
167 : public iterator_facade<join_iterator<Iterator1,Iterator2,ValueType,Reference,Traversal>, ValueType, Traversal, Reference>
|
Chris@16
|
168 {
|
Chris@16
|
169 typedef join_iterator_link<Iterator1, Iterator2> link_t;
|
Chris@16
|
170 typedef join_iterator_union<Iterator1, Iterator2, Reference> iterator_union;
|
Chris@16
|
171 public:
|
Chris@16
|
172 typedef Iterator1 iterator1_t;
|
Chris@16
|
173 typedef Iterator2 iterator2_t;
|
Chris@16
|
174
|
Chris@16
|
175 join_iterator()
|
Chris@16
|
176 : m_section(0u)
|
Chris@16
|
177 , m_it(0u, iterator1_t(), iterator2_t())
|
Chris@16
|
178 , m_link(link_t(iterator1_t(), iterator2_t()))
|
Chris@16
|
179 {}
|
Chris@16
|
180
|
Chris@16
|
181 join_iterator(unsigned int section, Iterator1 current1, Iterator1 last1, Iterator2 first2, Iterator2 current2)
|
Chris@16
|
182 : m_section(section)
|
Chris@16
|
183 , m_it(section, current1, current2)
|
Chris@16
|
184 , m_link(link_t(last1, first2))
|
Chris@16
|
185 {
|
Chris@16
|
186 }
|
Chris@16
|
187
|
Chris@16
|
188 template<typename Range1, typename Range2>
|
Chris@16
|
189 join_iterator(Range1& r1, Range2& r2, join_iterator_begin_tag)
|
Chris@16
|
190 : m_section(boost::empty(r1) ? 1u : 0u)
|
Chris@16
|
191 , m_it(boost::empty(r1) ? 1u : 0u, boost::begin(r1), boost::begin(r2))
|
Chris@16
|
192 , m_link(link_t(boost::end(r1), boost::begin(r2)))
|
Chris@16
|
193 {
|
Chris@16
|
194 }
|
Chris@16
|
195
|
Chris@16
|
196 template<typename Range1, typename Range2>
|
Chris@16
|
197 join_iterator(const Range1& r1, const Range2& r2, join_iterator_begin_tag)
|
Chris@16
|
198 : m_section(boost::empty(r1) ? 1u : 0u)
|
Chris@16
|
199 , m_it(boost::empty(r1) ? 1u : 0u, boost::const_begin(r1), boost::const_begin(r2))
|
Chris@16
|
200 , m_link(link_t(boost::const_end(r1), boost::const_begin(r2)))
|
Chris@16
|
201 {
|
Chris@16
|
202 }
|
Chris@16
|
203
|
Chris@16
|
204 template<typename Range1, typename Range2>
|
Chris@16
|
205 join_iterator(Range1& r1, Range2& r2, join_iterator_end_tag)
|
Chris@16
|
206 : m_section(1u)
|
Chris@16
|
207 , m_it(1u, boost::end(r1), boost::end(r2))
|
Chris@16
|
208 , m_link(link_t(boost::end(r1), boost::begin(r2)))
|
Chris@16
|
209 {
|
Chris@16
|
210 }
|
Chris@16
|
211
|
Chris@16
|
212 template<typename Range1, typename Range2>
|
Chris@16
|
213 join_iterator(const Range1& r1, const Range2& r2, join_iterator_end_tag)
|
Chris@16
|
214 : m_section(1u)
|
Chris@16
|
215 , m_it(1u, boost::const_end(r1), boost::const_end(r2))
|
Chris@16
|
216 , m_link(link_t(boost::const_end(r1), boost::const_begin(r2)))
|
Chris@16
|
217 {
|
Chris@16
|
218 }
|
Chris@16
|
219
|
Chris@16
|
220 private:
|
Chris@16
|
221 void increment()
|
Chris@16
|
222 {
|
Chris@16
|
223 if (m_section)
|
Chris@16
|
224 ++m_it.it2();
|
Chris@16
|
225 else
|
Chris@16
|
226 {
|
Chris@16
|
227 ++m_it.it1();
|
Chris@16
|
228 if (m_it.it1() == m_link.last1)
|
Chris@16
|
229 {
|
Chris@16
|
230 m_it.it2() = m_link.first2;
|
Chris@16
|
231 m_section = 1u;
|
Chris@16
|
232 }
|
Chris@16
|
233 }
|
Chris@16
|
234 }
|
Chris@16
|
235
|
Chris@16
|
236 void decrement()
|
Chris@16
|
237 {
|
Chris@16
|
238 if (m_section)
|
Chris@16
|
239 {
|
Chris@16
|
240 if (m_it.it2() == m_link.first2)
|
Chris@16
|
241 {
|
Chris@16
|
242 m_it.it1() = boost::prior(m_link.last1);
|
Chris@16
|
243 m_section = 0u;
|
Chris@16
|
244 }
|
Chris@16
|
245 else
|
Chris@16
|
246 --m_it.it2();
|
Chris@16
|
247 }
|
Chris@16
|
248 else
|
Chris@16
|
249 --m_it.it1();
|
Chris@16
|
250 }
|
Chris@16
|
251
|
Chris@16
|
252 typename join_iterator::reference dereference() const
|
Chris@16
|
253 {
|
Chris@16
|
254 return m_it.dereference(m_section);
|
Chris@16
|
255 }
|
Chris@16
|
256
|
Chris@16
|
257 bool equal(const join_iterator& other) const
|
Chris@16
|
258 {
|
Chris@16
|
259 return m_section == other.m_section
|
Chris@16
|
260 && m_it.equal(other.m_it, m_section);
|
Chris@16
|
261 }
|
Chris@16
|
262
|
Chris@16
|
263 void advance(typename join_iterator::difference_type offset)
|
Chris@16
|
264 {
|
Chris@16
|
265 if (m_section)
|
Chris@16
|
266 advance_from_range2(offset);
|
Chris@16
|
267 else
|
Chris@16
|
268 advance_from_range1(offset);
|
Chris@16
|
269 }
|
Chris@16
|
270
|
Chris@16
|
271 typename join_iterator::difference_type distance_to(const join_iterator& other) const
|
Chris@16
|
272 {
|
Chris@16
|
273 typename join_iterator::difference_type result;
|
Chris@16
|
274 if (m_section)
|
Chris@16
|
275 {
|
Chris@16
|
276 if (other.m_section)
|
Chris@16
|
277 result = other.m_it.it2() - m_it.it2();
|
Chris@16
|
278 else
|
Chris@16
|
279 {
|
Chris@16
|
280 result = (m_link.first2 - m_it.it2())
|
Chris@16
|
281 + (other.m_it.it1() - m_link.last1);
|
Chris@16
|
282
|
Chris@16
|
283 BOOST_ASSERT( result <= 0 );
|
Chris@16
|
284 }
|
Chris@16
|
285 }
|
Chris@16
|
286 else
|
Chris@16
|
287 {
|
Chris@16
|
288 if (other.m_section)
|
Chris@16
|
289 {
|
Chris@16
|
290 result = (m_link.last1 - m_it.it1())
|
Chris@16
|
291 + (other.m_it.it2() - m_link.first2);
|
Chris@16
|
292 }
|
Chris@16
|
293 else
|
Chris@16
|
294 result = other.m_it.it1() - m_it.it1();
|
Chris@16
|
295 }
|
Chris@16
|
296 return result;
|
Chris@16
|
297 }
|
Chris@16
|
298
|
Chris@16
|
299 void advance_from_range2(typename join_iterator::difference_type offset)
|
Chris@16
|
300 {
|
Chris@16
|
301 typedef typename join_iterator::difference_type difference_t;
|
Chris@16
|
302 BOOST_ASSERT( m_section == 1u );
|
Chris@16
|
303 if (offset < 0)
|
Chris@16
|
304 {
|
Chris@16
|
305 difference_t r2_dist = m_link.first2 - m_it.it2();
|
Chris@16
|
306 BOOST_ASSERT( r2_dist <= 0 );
|
Chris@16
|
307 if (offset >= r2_dist)
|
Chris@16
|
308 std::advance(m_it.it2(), offset);
|
Chris@16
|
309 else
|
Chris@16
|
310 {
|
Chris@16
|
311 difference_t r1_dist = offset - r2_dist;
|
Chris@16
|
312 BOOST_ASSERT( r1_dist <= 0 );
|
Chris@16
|
313 m_it.it1() = m_link.last1 + r1_dist;
|
Chris@16
|
314 m_section = 0u;
|
Chris@16
|
315 }
|
Chris@16
|
316 }
|
Chris@16
|
317 else
|
Chris@16
|
318 std::advance(m_it.it2(), offset);
|
Chris@16
|
319 }
|
Chris@16
|
320
|
Chris@16
|
321 void advance_from_range1(typename join_iterator::difference_type offset)
|
Chris@16
|
322 {
|
Chris@16
|
323 typedef typename join_iterator::difference_type difference_t;
|
Chris@16
|
324 BOOST_ASSERT( m_section == 0u );
|
Chris@16
|
325 if (offset > 0)
|
Chris@16
|
326 {
|
Chris@16
|
327 difference_t r1_dist = m_link.last1 - m_it.it1();
|
Chris@16
|
328 BOOST_ASSERT( r1_dist >= 0 );
|
Chris@16
|
329 if (offset < r1_dist)
|
Chris@16
|
330 std::advance(m_it.it1(), offset);
|
Chris@16
|
331 else
|
Chris@16
|
332 {
|
Chris@16
|
333 difference_t r2_dist = offset - r1_dist;
|
Chris@16
|
334 BOOST_ASSERT( r2_dist >= 0 );
|
Chris@16
|
335 m_it.it2() = m_link.first2 + r2_dist;
|
Chris@16
|
336 m_section = 1u;
|
Chris@16
|
337 }
|
Chris@16
|
338 }
|
Chris@16
|
339 else
|
Chris@16
|
340 std::advance(m_it.it1(), offset);
|
Chris@16
|
341 }
|
Chris@16
|
342
|
Chris@16
|
343 unsigned int m_section;
|
Chris@16
|
344 iterator_union m_it;
|
Chris@16
|
345 link_t m_link;
|
Chris@16
|
346
|
Chris@16
|
347 friend class ::boost::iterator_core_access;
|
Chris@16
|
348 };
|
Chris@16
|
349
|
Chris@16
|
350 } // namespace range_detail
|
Chris@16
|
351
|
Chris@16
|
352 } // namespace boost
|
Chris@16
|
353
|
Chris@16
|
354 #endif // include guard
|