#ifndef SPROUT_ALGORITHM_FIXED_REMOVE_COPY_HPP #define SPROUT_ALGORITHM_FIXED_REMOVE_COPY_HPP #include #include #include #include namespace sprout { namespace fixed { namespace detail { template SPROUT_CONSTEXPR inline typename std::enable_if< sprout::fixed_container_traits::fixed_size == sizeof...(Args), typename sprout::fixed_container_traits::fixed_container_type >::type remove_copy_impl_3( Result const& result, Args const&... args ) { return typename sprout::fixed_container_traits::fixed_container_type{args...}; } template SPROUT_CONSTEXPR inline typename std::enable_if< sprout::fixed_container_traits::fixed_size != sizeof...(Args), typename sprout::fixed_container_traits::fixed_container_type >::type remove_copy_impl_3( Result const& result, Args const&... args ) { return remove_copy_impl_3(result, args..., *(sprout::fixed_begin(result) + sizeof...(Args))); } template SPROUT_CONSTEXPR inline typename std::enable_if< sprout::fixed_container_traits::fixed_size == sizeof...(Args), typename sprout::fixed_container_traits::fixed_container_type >::type remove_copy_impl_2( Iterator first, Iterator last, Result const& result, T const& value, typename sprout::fixed_container_traits::difference_type offset, Args const&... args ) { return typename sprout::fixed_container_traits::fixed_container_type{args...}; } template SPROUT_CONSTEXPR inline typename std::enable_if< sprout::fixed_container_traits::fixed_size != sizeof...(Args), typename sprout::fixed_container_traits::fixed_container_type >::type remove_copy_impl_2( Iterator first, Iterator last, Result const& result, T const& value, typename sprout::fixed_container_traits::difference_type offset, Args const&... args ) { return first != last && sizeof...(Args) < offset ? *first == value ? remove_copy_impl_2(first + 1, last, result, value, offset, args...) : remove_copy_impl_2(first + 1, last, result, value, offset, args..., *first) : remove_copy_impl_3(result, args...) ; } template SPROUT_CONSTEXPR inline typename std::enable_if< sprout::fixed_container_traits::fixed_size == sizeof...(Args), typename sprout::fixed_container_traits::fixed_container_type >::type remove_copy_impl_1( Iterator first, Iterator last, Result const& result, T const& value, typename sprout::fixed_container_traits::difference_type offset, Args const&... args ) { return typename sprout::fixed_container_traits::fixed_container_type{args...}; } template SPROUT_CONSTEXPR inline typename std::enable_if< sprout::fixed_container_traits::fixed_size != sizeof...(Args), typename sprout::fixed_container_traits::fixed_container_type >::type remove_copy_impl_1( Iterator first, Iterator last, Result const& result, T const& value, typename sprout::fixed_container_traits::difference_type offset, Args const&... args ) { return sizeof...(Args) < offset ? remove_copy_impl_1(first, last, result, value, offset, args..., *(sprout::fixed_begin(result) + sizeof...(Args))) : remove_copy_impl_2(first, last, result, value, offset + sprout::size(result), args...) ; } template SPROUT_CONSTEXPR inline typename sprout::fixed_container_traits::fixed_container_type remove_copy_impl( Iterator first, Iterator last, Result const& result, T const& value ) { return remove_copy_impl_1(first, last, result, value, sprout::fixed_begin_offset(result)); } } // namespace detail // // remove_copy // template SPROUT_CONSTEXPR inline typename sprout::fixed_container_traits::fixed_container_type remove_copy( Iterator first, Iterator last, Result const& result, T const& value ) { return sprout::fixed::detail::remove_copy_impl(first, last, result, value); } } // namespace fixed } // namespace sprout #endif // #ifndef SPROUT_ALGORITHM_FIXED_REMOVE_COPY_HPP