#ifndef SPROUT_FUNCTIONAL_BIND1ST_HPP #define SPROUT_FUNCTIONAL_BIND1ST_HPP #include #include #include #include namespace sprout { // Copyright (C) 2011 RiSK (sscrisk) // D.9.1 Class template binder1st namespace detail { template class binder1st; template class binder1st< Fn, T, typename std::enable_if::value>::type > : public sprout::unary_function { public: typedef typename std::conditional< std::is_void::value, typename Fn::first_argument_type, T >::type value_type; protected: Fn op; value_type value; public: SPROUT_CONSTEXPR binder1st(Fn const& x, value_type const& y) : op(x), value(y) {} SPROUT_CONSTEXPR typename Fn::result_type operator()(typename Fn::second_argument_type const& x) const { return op(value, x); } }; template class binder1st< Fn, T, typename std::enable_if::value>::type > { public: typedef T value_type; protected: Fn op; value_type value; public: SPROUT_CONSTEXPR binder1st(Fn const& x, value_type const& y) : op(x), value(y) {} template SPROUT_CONSTEXPR decltype(op(value, std::declval())) operator()(U const& x) const { return op(value, x); } }; } // namespace detail template class binder1st : public sprout::detail::binder1st { public: typedef typename sprout::detail::binder1st::value_type value_type; public: SPROUT_CONSTEXPR binder1st(Fn const& x, value_type const& y) : sprout::detail::binder1st(x, y) {} }; // D.9.2 bind1st template inline SPROUT_CONSTEXPR sprout::binder1st bind1st(Fn const& fn, T const& x) { return sprout::binder1st(fn, typename sprout::binder1st::value_type(x)); } } // namespace sprout #endif // #ifndef SPROUT_FUNCTIONAL_BIND1ST_HPP