for_each_field_impl.hpp 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566
  1. // Copyright (c) 2016-2024 Antony Polukhin
  2. //
  3. // Distributed under the Boost Software License, Version 1.0. (See accompanying
  4. // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  5. #ifndef BOOST_PFR_DETAIL_FOR_EACH_FIELD_IMPL_HPP
  6. #define BOOST_PFR_DETAIL_FOR_EACH_FIELD_IMPL_HPP
  7. #pragma once
  8. #include <boost/pfr/detail/config.hpp>
  9. #include <utility> // metaprogramming stuff
  10. #include <boost/pfr/detail/sequence_tuple.hpp>
  11. #include <boost/pfr/detail/rvalue_t.hpp>
  12. namespace boost { namespace pfr { namespace detail {
  13. template <std::size_t Index>
  14. using size_t_ = std::integral_constant<std::size_t, Index >;
  15. template <class T, class F, class I, class = decltype(std::declval<F>()(std::declval<T>(), I{}))>
  16. constexpr void for_each_field_impl_apply(T&& v, F&& f, I i, long) {
  17. std::forward<F>(f)(std::forward<T>(v), i);
  18. }
  19. template <class T, class F, class I>
  20. constexpr void for_each_field_impl_apply(T&& v, F&& f, I /*i*/, int) {
  21. std::forward<F>(f)(std::forward<T>(v));
  22. }
  23. #if !defined(__cpp_fold_expressions) || __cpp_fold_expressions < 201603
  24. template <class T, class F, std::size_t... I>
  25. constexpr void for_each_field_impl(T& t, F&& f, std::index_sequence<I...>, std::false_type /*move_values*/) {
  26. const int v[] = {0, (
  27. detail::for_each_field_impl_apply(sequence_tuple::get<I>(t), std::forward<F>(f), size_t_<I>{}, 1L),
  28. 0
  29. )...};
  30. (void)v;
  31. }
  32. template <class T, class F, std::size_t... I>
  33. constexpr void for_each_field_impl(T& t, F&& f, std::index_sequence<I...>, std::true_type /*move_values*/) {
  34. const int v[] = {0, (
  35. detail::for_each_field_impl_apply(sequence_tuple::get<I>(std::move(t)), std::forward<F>(f), size_t_<I>{}, 1L),
  36. 0
  37. )...};
  38. (void)v;
  39. }
  40. #else
  41. template <class T, class F, std::size_t... I>
  42. constexpr void for_each_field_impl(T& t, F&& f, std::index_sequence<I...>, std::false_type /*move_values*/) {
  43. (detail::for_each_field_impl_apply(sequence_tuple::get<I>(t), std::forward<F>(f), size_t_<I>{}, 1L), ...);
  44. }
  45. template <class T, class F, std::size_t... I>
  46. constexpr void for_each_field_impl(T& t, F&& f, std::index_sequence<I...>, std::true_type /*move_values*/) {
  47. (detail::for_each_field_impl_apply(sequence_tuple::get<I>(std::move(t)), std::forward<F>(f), size_t_<I>{}, 1L), ...);
  48. }
  49. #endif
  50. }}} // namespace boost::pfr::detail
  51. #endif // BOOST_PFR_DETAIL_FOR_EACH_FIELD_IMPL_HPP