traits.h 1.2 KB

1234567891011121314151617181920212223242526272829303132333435363738
  1. #pragma once
  2. #define _val(type) std::declval<type>()
  3. #define exists(expr) std::void_t<decltype(expr)>
  4. #include <iterator/iterator_fwd.hpp>
  5. namespace iterator::detail {
  6. template <typename T, typename = void> struct reference_helper {
  7. using type = decltype(*std::declval<T>());
  8. };
  9. template <typename T>
  10. struct reference_helper<T, std::void_t<typename T::reference>> {
  11. using type = typename T::reference;
  12. };
  13. template <typename T, typename = void> struct value_type_helper {
  14. using reference = typename reference_helper<T>::type;
  15. using type = std::remove_cv_t<std::remove_reference_t<reference>>;
  16. };
  17. template <typename T>
  18. struct value_type_helper<T, std::void_t<typename T::value_type>> {
  19. using type = typename T::value_type;
  20. };
  21. template <typename T> using value_type = typename value_type_helper<T>::type;
  22. template <typename T> using reference = typename reference_helper<T>::type;
  23. template <typename C, typename = void> struct is_container : std::false_type {};
  24. template <typename C>
  25. struct is_container<C, std::void_t<iter<C>>> : std::true_type {};
  26. template <typename C> constexpr bool is_container_v = is_container<C>{};
  27. template <typename Iter>
  28. constexpr bool is_rvalue_iterator_v =
  29. !std::is_reference_v<decltype(*std::declval<Iter>())>;
  30. }