2016-08-18 22:34:36 +08:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "utils/iterator/iterator_base.hpp"
|
|
|
|
#include "utils/option.hpp"
|
|
|
|
|
|
|
|
namespace iter
|
|
|
|
{
|
|
|
|
|
|
|
|
// Class which turns accessor int next() based iterator.
|
|
|
|
// T - type of return value
|
|
|
|
// I - iterator type gotten from accessor
|
|
|
|
// A - accessor type
|
|
|
|
template <class T, class I, class A>
|
|
|
|
class IteratorAccessor : public IteratorBase<T>
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
IteratorAccessor() = delete;
|
|
|
|
|
|
|
|
IteratorAccessor(A &&acc)
|
2016-08-30 12:29:30 +08:00
|
|
|
: begin(std::move(acc.begin())), acc(std::forward<A>(acc)), returned(0)
|
2016-08-18 22:34:36 +08:00
|
|
|
{
|
|
|
|
}
|
2016-08-28 22:47:13 +08:00
|
|
|
|
|
|
|
IteratorAccessor(IteratorAccessor &&other)
|
2016-08-30 12:29:30 +08:00
|
|
|
: begin(std::move(other.begin)), acc(std::forward<A>(other.acc)),
|
|
|
|
returned(other.returned)
|
2016-08-28 22:47:13 +08:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
~IteratorAccessor() final {}
|
|
|
|
|
2016-08-18 22:34:36 +08:00
|
|
|
// Iter(const Iter &other) = delete;
|
|
|
|
// Iter(Iter &&other) :
|
|
|
|
// begin(std::move(other.begin)),end(std::move(other.end)) {};
|
|
|
|
|
|
|
|
Option<T> next() final
|
|
|
|
{
|
|
|
|
if (begin != acc.end()) {
|
|
|
|
auto ret = Option<T>(&(*(begin.operator->())));
|
|
|
|
begin++;
|
2016-08-30 12:29:30 +08:00
|
|
|
returned++;
|
2016-08-18 22:34:36 +08:00
|
|
|
return ret;
|
|
|
|
} else {
|
|
|
|
return Option<T>();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-30 12:29:30 +08:00
|
|
|
Count count() final
|
|
|
|
{
|
|
|
|
auto size = acc.size();
|
|
|
|
if (size > returned) {
|
|
|
|
return Count(0);
|
|
|
|
} else {
|
|
|
|
return Count(size - returned);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-18 22:34:36 +08:00
|
|
|
private:
|
|
|
|
I begin;
|
|
|
|
A acc;
|
2016-08-30 12:29:30 +08:00
|
|
|
size_t returned;
|
2016-08-18 22:34:36 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
// TODO: Join to make functions into one
|
|
|
|
template <class A>
|
|
|
|
auto make_iter(A &&acc)
|
|
|
|
{
|
|
|
|
// Compiler cant deduce types T and I. decltype are here to help with it.
|
|
|
|
return IteratorAccessor<decltype(&(*(acc.begin().operator->()))),
|
|
|
|
decltype(acc.begin()), A>(std::move(acc));
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class A>
|
|
|
|
auto make_iter_ref(A &acc)
|
|
|
|
{
|
|
|
|
// Compiler cant deduce types T and I. decltype are here to help with it.
|
|
|
|
return IteratorAccessor<decltype(&(*(acc.begin().operator->()))),
|
|
|
|
decltype(acc.begin()), A &>(acc);
|
|
|
|
}
|
|
|
|
}
|