89 lines
2.7 KiB
C++
89 lines
2.7 KiB
C++
/*!
|
|
@file
|
|
Defines `boost::hana::overload`.
|
|
|
|
@copyright Louis Dionne 2013-2016
|
|
Distributed under the Boost Software License, Version 1.0.
|
|
(See accompanying file LICENSE.md or copy at http://boost.org/LICENSE_1_0.txt)
|
|
*/
|
|
|
|
#ifndef BOOST_HANA_FUNCTIONAL_OVERLOAD_HPP
|
|
#define BOOST_HANA_FUNCTIONAL_OVERLOAD_HPP
|
|
|
|
#include <boost/hana/config.hpp>
|
|
#include <boost/hana/detail/decay.hpp>
|
|
|
|
|
|
BOOST_HANA_NAMESPACE_BEGIN
|
|
//! @ingroup group-functional
|
|
//! Pick one of several functions to call based on overload resolution.
|
|
//!
|
|
//! Specifically, `overload(f1, f2, ..., fn)` is a function object such
|
|
//! that
|
|
//! @code
|
|
//! overload(f1, f2, ..., fn)(x...) == fk(x...)
|
|
//! @endcode
|
|
//!
|
|
//! where `fk` is the function of `f1, ..., fn` that would be called if
|
|
//! overload resolution was performed amongst that set of functions only.
|
|
//! If more than one function `fk` would be picked by overload resolution,
|
|
//! then the call is ambiguous.
|
|
//!
|
|
//! ### Example
|
|
//! @include example/functional/overload.cpp
|
|
#ifdef BOOST_HANA_DOXYGEN_INVOKED
|
|
constexpr auto overload = [](auto&& f1, auto&& f2, ..., auto&& fn) {
|
|
return [perfect-capture](auto&& ...x) -> decltype(auto) {
|
|
return forwarded(fk)(forwarded(x)...);
|
|
};
|
|
};
|
|
#else
|
|
template <typename F, typename ...G>
|
|
struct overload_t
|
|
: overload_t<F>::type
|
|
, overload_t<G...>::type
|
|
{
|
|
using type = overload_t;
|
|
using overload_t<F>::type::operator();
|
|
using overload_t<G...>::type::operator();
|
|
|
|
template <typename F_, typename ...G_>
|
|
constexpr explicit overload_t(F_&& f, G_&& ...g)
|
|
: overload_t<F>::type(static_cast<F_&&>(f))
|
|
, overload_t<G...>::type(static_cast<G_&&>(g)...)
|
|
{ }
|
|
};
|
|
|
|
template <typename F>
|
|
struct overload_t<F> { using type = F; };
|
|
|
|
template <typename R, typename ...Args>
|
|
struct overload_t<R(*)(Args...)> {
|
|
using type = overload_t;
|
|
R (*fptr_)(Args...);
|
|
|
|
explicit constexpr overload_t(R (*fp)(Args...))
|
|
: fptr_(fp)
|
|
{ }
|
|
|
|
constexpr R operator()(Args ...args) const
|
|
{ return fptr_(static_cast<Args&&>(args)...); }
|
|
};
|
|
|
|
struct make_overload_t {
|
|
template <typename ...F,
|
|
typename Overload = typename overload_t<
|
|
typename detail::decay<F>::type...
|
|
>::type
|
|
>
|
|
constexpr Overload operator()(F&& ...f) const {
|
|
return Overload(static_cast<F&&>(f)...);
|
|
}
|
|
};
|
|
|
|
constexpr make_overload_t overload{};
|
|
#endif
|
|
BOOST_HANA_NAMESPACE_END
|
|
|
|
#endif // !BOOST_HANA_FUNCTIONAL_OVERLOAD_HPP
|