shuffle.hpp 2.6 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283
  1. /// \file
  2. // Range v3 library
  3. //
  4. // Copyright Filip Matzner 2015
  5. //
  6. // Use, modification and distribution is subject to the
  7. // Boost Software License, Version 1.0. (See accompanying
  8. // file LICENSE_1_0.txt or copy at
  9. // http://www.boost.org/LICENSE_1_0.txt)
  10. //
  11. // Project home: https://github.com/ericniebler/range-v3
  12. //
  13. #ifndef RANGES_V3_ACTION_SHUFFLE_HPP
  14. #define RANGES_V3_ACTION_SHUFFLE_HPP
  15. #include <range/v3/range_fwd.hpp>
  16. #include <range/v3/action/action.hpp>
  17. #include <range/v3/algorithm/shuffle.hpp>
  18. #include <range/v3/functional/bind.hpp>
  19. #include <range/v3/functional/bind_back.hpp>
  20. #include <range/v3/functional/invoke.hpp>
  21. #include <range/v3/iterator/concepts.hpp>
  22. #include <range/v3/iterator/traits.hpp>
  23. #include <range/v3/utility/static_const.hpp>
  24. #include <range/v3/detail/prologue.hpp>
  25. namespace ranges
  26. {
  27. /// \addtogroup group-actions
  28. /// @{
  29. namespace actions
  30. {
  31. struct shuffle_fn
  32. {
  33. template(typename Gen)(
  34. requires uniform_random_bit_generator<Gen>)
  35. constexpr auto operator()(Gen & gen) const
  36. {
  37. return make_action_closure(
  38. bind_back(shuffle_fn{}, detail::reference_wrapper_<Gen>(gen)));
  39. }
  40. template(typename Gen)(
  41. requires uniform_random_bit_generator<Gen>)
  42. constexpr auto operator()(Gen && gen) const
  43. {
  44. return make_action_closure(
  45. bind_back(shuffle_fn{}, static_cast<Gen &&>(gen)));
  46. }
  47. template(typename Rng, typename Gen)(
  48. requires random_access_range<Rng> AND permutable<iterator_t<Rng>> AND
  49. uniform_random_bit_generator<std::remove_reference_t<Gen>> AND
  50. convertible_to<invoke_result_t<Gen &>, range_difference_t<Rng>>)
  51. Rng operator()(Rng && rng, Gen && gen) const
  52. {
  53. ranges::shuffle(rng, static_cast<Gen &&>(gen));
  54. return static_cast<Rng &&>(rng);
  55. }
  56. /// \cond
  57. template<typename Rng, typename T>
  58. invoke_result_t<shuffle_fn, Rng, T &> //
  59. operator()(Rng && rng, detail::reference_wrapper_<T> r) const
  60. {
  61. return (*this)(static_cast<Rng &&>(rng), r.get());
  62. }
  63. /// \endcond
  64. };
  65. /// \relates actions::shuffle_fn
  66. /// \sa `action_closure`
  67. RANGES_INLINE_VARIABLE(shuffle_fn, shuffle)
  68. } // namespace actions
  69. /// @}
  70. } // namespace ranges
  71. #include <range/v3/detail/epilogue.hpp>
  72. #endif