2016-08-18 22:34:36 +08:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "utils/option.hpp"
|
|
|
|
|
|
|
|
namespace iter
|
|
|
|
{
|
|
|
|
|
2016-10-19 08:41:06 +08:00
|
|
|
// Class which wraps iterator with next() into C++ iterator.
|
2016-08-18 22:34:36 +08:00
|
|
|
// T - type of return value
|
|
|
|
// I - iterator type
|
|
|
|
template <class T, class I>
|
|
|
|
class RangeIterator
|
|
|
|
{
|
|
|
|
|
|
|
|
public:
|
|
|
|
RangeIterator() : iter(Option<I>()), value(Option<T>()){};
|
|
|
|
|
|
|
|
RangeIterator(I &&iter)
|
|
|
|
: value(iter.next()), iter(Option<I>(std::move(iter)))
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
T &operator*()
|
|
|
|
{
|
|
|
|
assert(value.is_present());
|
|
|
|
return value.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
T *operator->()
|
|
|
|
{
|
|
|
|
assert(value.is_present());
|
|
|
|
return &value.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
operator T &()
|
|
|
|
{
|
|
|
|
assert(value.is_present());
|
|
|
|
return value.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
RangeIterator &operator++()
|
|
|
|
{
|
|
|
|
assert(iter.is_present());
|
|
|
|
value = iter.get().next();
|
|
|
|
return (*this);
|
|
|
|
}
|
|
|
|
|
|
|
|
RangeIterator &operator++(int) { return operator++(); }
|
|
|
|
|
|
|
|
friend bool operator==(const RangeIterator &a, const RangeIterator &b)
|
|
|
|
{
|
|
|
|
return a.value.is_present() == b.value.is_present();
|
|
|
|
}
|
|
|
|
|
|
|
|
friend bool operator!=(const RangeIterator &a, const RangeIterator &b)
|
|
|
|
{
|
|
|
|
return !(a == b);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
Option<I> iter;
|
|
|
|
Option<T> value;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <class I>
|
|
|
|
auto make_range_iterator(I &&iter)
|
|
|
|
{
|
2016-10-19 08:41:06 +08:00
|
|
|
// Because this function isn't receving or in any way using type T from
|
|
|
|
// RangeIterator, compiler can't deduce it. That is the reason why
|
|
|
|
// there is decltype in construction of RangeIterator.
|
|
|
|
// declytype(iter.next().take()) is T.
|
2016-08-18 22:34:36 +08:00
|
|
|
return RangeIterator<decltype(iter.next().take()), I>(std::move(iter));
|
|
|
|
}
|
|
|
|
}
|