clamp.hpp 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. /*
  2. Copyright (c) Marshall Clow 2008-2012.
  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. Revision history:
  6. 27 June 2009 mtc First version
  7. 23 Oct 2010 mtc Added predicate version
  8. */
  9. /// \file clamp.hpp
  10. /// \brief Clamp algorithm
  11. /// \author Marshall Clow
  12. ///
  13. /// Suggested by olafvdspek in https://svn.boost.org/trac/boost/ticket/3215
  14. #ifndef BOOST_ALGORITHM_CLAMP_HPP
  15. #define BOOST_ALGORITHM_CLAMP_HPP
  16. #include <functional> // For std::less
  17. #include <iterator> // For std::iterator_traits
  18. #include <cassert>
  19. #include <boost/range/begin.hpp>
  20. #include <boost/range/end.hpp>
  21. #include <boost/mpl/identity.hpp> // for identity
  22. #include <boost/utility/enable_if.hpp> // for boost::disable_if
  23. namespace boost { namespace algorithm {
  24. /// \fn clamp ( T const& val,
  25. /// typename boost::mpl::identity<T>::type const & lo,
  26. /// typename boost::mpl::identity<T>::type const & hi, Pred p )
  27. /// \return the value "val" brought into the range [ lo, hi ]
  28. /// using the comparison predicate p.
  29. /// If p ( val, lo ) return lo.
  30. /// If p ( hi, val ) return hi.
  31. /// Otherwise, return the original value.
  32. ///
  33. /// \param val The value to be clamped
  34. /// \param lo The lower bound of the range to be clamped to
  35. /// \param hi The upper bound of the range to be clamped to
  36. /// \param p A predicate to use to compare the values.
  37. /// p ( a, b ) returns a boolean.
  38. ///
  39. template<typename T, typename Pred>
  40. T const & clamp ( T const& val,
  41. typename boost::mpl::identity<T>::type const & lo,
  42. typename boost::mpl::identity<T>::type const & hi, Pred p )
  43. {
  44. // assert ( !p ( hi, lo )); // Can't assert p ( lo, hi ) b/c they might be equal
  45. return p ( val, lo ) ? lo : p ( hi, val ) ? hi : val;
  46. }
  47. /// \fn clamp ( T const& val,
  48. /// typename boost::mpl::identity<T>::type const & lo,
  49. /// typename boost::mpl::identity<T>::type const & hi )
  50. /// \return the value "val" brought into the range [ lo, hi ].
  51. /// If the value is less than lo, return lo.
  52. /// If the value is greater than "hi", return hi.
  53. /// Otherwise, return the original value.
  54. ///
  55. /// \param val The value to be clamped
  56. /// \param lo The lower bound of the range to be clamped to
  57. /// \param hi The upper bound of the range to be clamped to
  58. ///
  59. template<typename T>
  60. T const& clamp ( const T& val,
  61. typename boost::mpl::identity<T>::type const & lo,
  62. typename boost::mpl::identity<T>::type const & hi )
  63. {
  64. return (clamp) ( val, lo, hi, std::less<T>());
  65. }
  66. /// \fn clamp_range ( InputIterator first, InputIterator last, OutputIterator out,
  67. /// std::iterator_traits<InputIterator>::value_type const & lo,
  68. /// std::iterator_traits<InputIterator>::value_type const & hi )
  69. /// \return clamp the sequence of values [first, last) into [ lo, hi ]
  70. ///
  71. /// \param first The start of the range of values
  72. /// \param last One past the end of the range of input values
  73. /// \param out An output iterator to write the clamped values into
  74. /// \param lo The lower bound of the range to be clamped to
  75. /// \param hi The upper bound of the range to be clamped to
  76. ///
  77. template<typename InputIterator, typename OutputIterator>
  78. OutputIterator clamp_range ( InputIterator first, InputIterator last, OutputIterator out,
  79. typename std::iterator_traits<InputIterator>::value_type const & lo,
  80. typename std::iterator_traits<InputIterator>::value_type const & hi )
  81. {
  82. // this could also be written with bind and std::transform
  83. while ( first != last )
  84. *out++ = clamp ( *first++, lo, hi );
  85. return out;
  86. }
  87. /// \fn clamp_range ( const Range &r, OutputIterator out,
  88. /// typename std::iterator_traits<typename boost::range_iterator<const Range>::type>::value_type const & lo,
  89. /// typename std::iterator_traits<typename boost::range_iterator<const Range>::type>::value_type const & hi )
  90. /// \return clamp the sequence of values [first, last) into [ lo, hi ]
  91. ///
  92. /// \param r The range of values to be clamped
  93. /// \param out An output iterator to write the clamped values into
  94. /// \param lo The lower bound of the range to be clamped to
  95. /// \param hi The upper bound of the range to be clamped to
  96. ///
  97. template<typename Range, typename OutputIterator>
  98. typename boost::disable_if_c<boost::is_same<Range, OutputIterator>::value, OutputIterator>::type
  99. clamp_range ( const Range &r, OutputIterator out,
  100. typename std::iterator_traits<typename boost::range_iterator<const Range>::type>::value_type const & lo,
  101. typename std::iterator_traits<typename boost::range_iterator<const Range>::type>::value_type const & hi )
  102. {
  103. return clamp_range ( boost::begin ( r ), boost::end ( r ), out, lo, hi );
  104. }
  105. /// \fn clamp_range ( InputIterator first, InputIterator last, OutputIterator out,
  106. /// std::iterator_traits<InputIterator>::value_type const & lo,
  107. /// std::iterator_traits<InputIterator>::value_type const & hi, Pred p )
  108. /// \return clamp the sequence of values [first, last) into [ lo, hi ]
  109. /// using the comparison predicate p.
  110. ///
  111. /// \param first The start of the range of values
  112. /// \param last One past the end of the range of input values
  113. /// \param out An output iterator to write the clamped values into
  114. /// \param lo The lower bound of the range to be clamped to
  115. /// \param hi The upper bound of the range to be clamped to
  116. /// \param p A predicate to use to compare the values.
  117. /// p ( a, b ) returns a boolean.
  118. ///
  119. template<typename InputIterator, typename OutputIterator, typename Pred>
  120. OutputIterator clamp_range ( InputIterator first, InputIterator last, OutputIterator out,
  121. typename std::iterator_traits<InputIterator>::value_type const & lo,
  122. typename std::iterator_traits<InputIterator>::value_type const & hi, Pred p )
  123. {
  124. // this could also be written with bind and std::transform
  125. while ( first != last )
  126. *out++ = clamp ( *first++, lo, hi, p );
  127. return out;
  128. }
  129. /// \fn clamp_range ( const Range &r, OutputIterator out,
  130. /// typename std::iterator_traits<typename boost::range_iterator<const Range>::type>::value_type const & lo,
  131. /// typename std::iterator_traits<typename boost::range_iterator<const Range>::type>::value_type const & hi,
  132. /// Pred p )
  133. /// \return clamp the sequence of values [first, last) into [ lo, hi ]
  134. /// using the comparison predicate p.
  135. ///
  136. /// \param r The range of values to be clamped
  137. /// \param out An output iterator to write the clamped values into
  138. /// \param lo The lower bound of the range to be clamped to
  139. /// \param hi The upper bound of the range to be clamped to
  140. /// \param p A predicate to use to compare the values.
  141. /// p ( a, b ) returns a boolean.
  142. //
  143. // Disable this template if the first two parameters are the same type;
  144. // In that case, the user will get the two iterator version.
  145. template<typename Range, typename OutputIterator, typename Pred>
  146. typename boost::disable_if_c<boost::is_same<Range, OutputIterator>::value, OutputIterator>::type
  147. clamp_range ( const Range &r, OutputIterator out,
  148. typename std::iterator_traits<typename boost::range_iterator<const Range>::type>::value_type const & lo,
  149. typename std::iterator_traits<typename boost::range_iterator<const Range>::type>::value_type const & hi,
  150. Pred p )
  151. {
  152. return clamp_range ( boost::begin ( r ), boost::end ( r ), out, lo, hi, p );
  153. }
  154. }}
  155. #endif // BOOST_ALGORITHM_CLAMP_HPP