Sprout/sprout/utility/pair/pair_decl.hpp

174 lines
5.6 KiB
C++
Raw Normal View History

2013-08-08 09:54:33 +00:00
/*=============================================================================
2014-01-08 07:48:12 +00:00
Copyright (c) 2011-2014 Bolero MURAKAMI
2013-08-08 09:54:33 +00:00
https://github.com/bolero-MURAKAMI/Sprout
Distributed under the Boost Software License, Version 1.0. (See accompanying
file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
=============================================================================*/
2013-03-27 08:55:14 +00:00
#ifndef SPROUT_UTILITY_PAIR_PAIR_DECL_HPP
#define SPROUT_UTILITY_PAIR_PAIR_DECL_HPP
2013-03-31 01:09:02 +00:00
#include <utility>
2013-03-27 08:55:14 +00:00
#include <type_traits>
#include <sprout/config.hpp>
2013-04-06 04:06:51 +00:00
#include <sprout/index_tuple/metafunction.hpp>
2013-03-27 08:55:14 +00:00
#include <sprout/utility/forward.hpp>
#include <sprout/utility/swap.hpp>
#include <sprout/utility/pair/pair_fwd.hpp>
#include <sprout/tuple/tuple/tuple_fwd.hpp>
namespace sprout {
//
// pair
//
template<typename T1, typename T2>
2013-03-27 08:55:14 +00:00
struct pair {
public:
typedef T1 first_type;
typedef T2 second_type;
public:
T1 first;
T2 second;
private:
template<typename... Args1, typename... Args2, sprout::index_t... Indexes1, sprout::index_t... Indexes2>
2013-03-27 08:55:14 +00:00
SPROUT_CONSTEXPR pair(
2013-04-05 09:11:56 +00:00
sprout::tuples::tuple<Args1...> first_args, sprout::tuples::tuple<Args2...> second_args,
sprout::index_tuple<Indexes1...>, sprout::index_tuple<Indexes2...>
2013-03-27 08:55:14 +00:00
);
public:
2013-03-31 09:20:59 +00:00
SPROUT_CONSTEXPR pair()
: first()
, second()
{}
2013-10-29 10:15:52 +00:00
pair(pair const&) = default;
pair(pair&&) = default;
2013-03-31 09:20:59 +00:00
SPROUT_CONSTEXPR pair(T1 const& x, T2 const& y)
: first(x)
, second(y)
{}
2013-03-31 01:09:02 +00:00
template<
typename U, typename V,
typename = typename std::enable_if<
std::is_constructible<first_type, U&&>::value && std::is_constructible<second_type, V&&>::value
>::type
>
2013-03-31 09:20:59 +00:00
SPROUT_CONSTEXPR pair(U&& x, V&& y)
: first(SPROUT_FORWARD(U, x))
, second(SPROUT_FORWARD(V, y))
2013-03-31 09:20:59 +00:00
{}
2013-03-31 01:09:02 +00:00
template<
typename U, typename V,
typename = typename std::enable_if<
std::is_constructible<first_type, U const&>::value && std::is_constructible<second_type, V const&>::value
>::type
>
2013-03-31 09:20:59 +00:00
SPROUT_CONSTEXPR pair(sprout::pair<U, V> const& other)
: first(other.first)
, second(other.second)
{}
2013-03-31 01:09:02 +00:00
template<
typename U, typename V,
typename = typename std::enable_if<
std::is_constructible<first_type, U&&>::value && std::is_constructible<second_type, V&&>::value
>::type
>
2013-03-31 09:20:59 +00:00
SPROUT_CONSTEXPR pair(sprout::pair<U, V>&& other)
: first(SPROUT_FORWARD(U, other.first))
, second(SPROUT_FORWARD(V, other.second))
2013-03-31 09:20:59 +00:00
{}
2013-03-31 01:09:02 +00:00
template<
typename... Args1, typename... Args2,
typename = typename std::enable_if<
std::is_constructible<first_type, Args1&&...>::value && std::is_constructible<second_type, Args2&&...>::value
>::type
>
2013-03-27 08:55:14 +00:00
SPROUT_CONSTEXPR pair(
2013-03-31 01:09:02 +00:00
std::piecewise_construct_t,
sprout::tuples::tuple<Args1...> first_args, sprout::tuples::tuple<Args2...> second_args
2013-03-27 08:55:14 +00:00
);
2013-03-31 01:09:02 +00:00
template<
typename U, typename V,
typename = typename std::enable_if<
std::is_constructible<first_type, U const&>::value && std::is_constructible<second_type, V const&>::value
>::type
>
SPROUT_CONSTEXPR pair(sprout::tuples::tuple<U, V> const& other);
template<
typename U, typename V,
typename = typename std::enable_if<
std::is_constructible<first_type, U&&>::value && std::is_constructible<second_type, V&&>::value
>::type
>
SPROUT_CONSTEXPR pair(sprout::tuples::tuple<U, V>&& other);
2013-10-29 10:15:52 +00:00
SPROUT_CXX14_CONSTEXPR pair& operator=(pair const& rhs) {
first = rhs.first;
second = rhs.second;
return *this;
}
SPROUT_CXX14_CONSTEXPR pair& operator=(pair&& rhs)
2013-03-31 09:20:59 +00:00
SPROUT_NOEXCEPT_EXPR(std::is_nothrow_move_assignable<T1>::value && std::is_nothrow_move_assignable<T2>::value)
{
first = SPROUT_FORWARD(T1, rhs.first);
second = SPROUT_FORWARD(T2, rhs.second);
2013-03-31 09:20:59 +00:00
return *this;
}
2013-03-31 01:09:02 +00:00
template<
typename U, typename V,
typename = typename std::enable_if<
std::is_assignable<first_type&, U const&>::value && std::is_assignable<second_type&, V const&>::value
>::type
>
SPROUT_CXX14_CONSTEXPR pair& operator=(sprout::pair<U, V> const& rhs) {
2013-03-31 09:20:59 +00:00
first = rhs.first;
second = rhs.second;
return *this;
}
2013-03-31 01:09:02 +00:00
template<
typename U, typename V,
typename = typename std::enable_if<
std::is_assignable<first_type&, U&&>::value && std::is_assignable<second_type&, V&&>::value
>::type
>
SPROUT_CXX14_CONSTEXPR pair& operator=(sprout::pair<U, V>&& rhs) {
first = SPROUT_FORWARD(U, rhs.first);
second = SPROUT_FORWARD(V, rhs.second);
2013-03-31 09:20:59 +00:00
return *this;
}
2013-03-31 01:09:02 +00:00
template<
typename U, typename V,
typename = typename std::enable_if<
std::is_assignable<first_type&, U const&>::value && std::is_assignable<second_type&, V const&>::value
>::type
>
SPROUT_CXX14_CONSTEXPR pair& operator=(sprout::tuples::tuple<U, V> const& rhs);
2013-03-31 01:09:02 +00:00
template<
typename U, typename V,
typename = typename std::enable_if<
std::is_assignable<first_type&, U&&>::value && std::is_assignable<second_type&, V&&>::value
>::type
>
SPROUT_CXX14_CONSTEXPR pair& operator=(sprout::tuples::tuple<U, V>&& rhs);
2013-03-31 01:09:02 +00:00
SPROUT_CXX14_CONSTEXPR void swap(pair& other)
2013-03-31 09:20:59 +00:00
SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(sprout::swap(first, other.first)) && SPROUT_NOEXCEPT_EXPR(sprout::swap(second, other.second))) {
sprout::swap(first, other.first);
sprout::swap(second, other.second);
}
2013-03-27 08:55:14 +00:00
};
//
// swap
//
template<typename T1, typename T2>
inline SPROUT_CXX14_CONSTEXPR void
2013-03-27 08:55:14 +00:00
swap(sprout::pair<T1, T2>& lhs, sprout::pair<T1, T2>& rhs)
2013-03-31 09:20:59 +00:00
SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(lhs.swap(rhs)))
{
lhs.swap(rhs);
}
2013-03-27 08:55:14 +00:00
} // namespace sprout
#endif // #ifndef SPROUT_UTILITY_PAIR_PAIR_DECL_HPP