map.hpp 1.4 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950
  1. #pragma once
  2. #include <iterator/proxy.h>
  3. namespace stream::detail {
  4. template <typename T, typename R>
  5. class map_iterator
  6. : public ::iterator::proxy<iterator<T>, map_iterator<T, R>> {
  7. public:
  8. using projection_t = std::function<R(T const &)>;
  9. using super_t = ::iterator::proxy<iterator<T>, map_iterator<T, R>>;
  10. public:
  11. map_iterator(projection_t proj, iterator<T> && impl)
  12. : proj_(std::move(proj)), super_t(std::move(impl)) {}
  13. R dereference() const { return proj_(super_t::dereference()); }
  14. private:
  15. std::function<R(T const &)> proj_;
  16. };
  17. template <typename T, typename R> class map_stream {
  18. public:
  19. template <typename F>
  20. map_stream(F && func, stream_base<T> const & sb)
  21. : fun_(func), source_(sb) {}
  22. iterator<R> begin() { return {map_iterator<T, R>{fun_, source_.begin()}}; }
  23. iterator<R> end() { return {map_iterator<T, R>{fun_, source_.end()}}; }
  24. private:
  25. std::function<R(T const &)> fun_;
  26. stream_base<T> source_;
  27. };
  28. template <typename T>
  29. template <typename F>
  30. stream_base<traits::mapped_t<T, F>> stream_base<T>::map(F && func) && {
  31. return std::make_shared<map_stream<T, traits::mapped_t<T, F>>>(
  32. func, std::move(*this));
  33. }
  34. template <typename T>
  35. template <typename F>
  36. stream_base<traits::mapped_t<T, F>> stream_base<T>::map(F && func) const & {
  37. return std::make_shared<map_stream<T, traits::mapped_t<T, F>>>(func, *this);
  38. }
  39. }