Sprout/sprout/generator/generator_access_traits.hpp

201 lines
7.5 KiB
C++
Raw Normal View History

2013-08-08 09:54:33 +00:00
/*=============================================================================
Copyright (c) 2011-2013 Bolero MURAKAMI
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)
=============================================================================*/
#ifndef SPROUT_GENERATOR_GENERATOR_ACCESS_TRAITS_HPP
#define SPROUT_GENERATOR_GENERATOR_ACCESS_TRAITS_HPP
#include <utility>
#include <type_traits>
#include <sprout/config.hpp>
#include <sprout/tuple/tuple.hpp>
#include <sprout/type_traits/integral_constant.hpp>
namespace sprout {
namespace generators {
namespace detail {
template<typename T>
struct has_mem_generated_value_test {
public:
template<
typename U = T,
typename = decltype(std::declval<U>().generated_value())
>
static sprout::true_type test(int);
static sprout::false_type test(...);
};
#if defined(_MSC_VER)
template<typename T, typename Base_ = decltype(sprout::generators::detail::has_mem_generated_value_test<T>::test(0))>
struct has_mem_generated_value
: public Base_
{};
#else
template<typename T>
struct has_mem_generated_value
: public decltype(sprout::generators::detail::has_mem_generated_value_test<T>::test(0))
{};
#endif
template<typename Gen, typename = void>
struct generator_access_traits_generated_value_impl;
template<typename Gen>
struct generator_access_traits_generated_value_impl<
Gen,
typename std::enable_if<sprout::generators::detail::has_mem_generated_value<Gen>::value>::type
> {
public:
static SPROUT_CONSTEXPR decltype(std::declval<Gen&>().generated_value())
2013-04-22 04:35:01 +00:00
get_generated_value(Gen& t)
2013-04-22 14:55:30 +00:00
SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(std::declval<Gen&>().generated_value()))
{
return t.generated_value();
}
static SPROUT_CONSTEXPR decltype(std::declval<Gen&&>().generated_value())
2013-04-22 04:35:01 +00:00
get_generated_value(Gen&& t)
2013-04-22 14:55:30 +00:00
SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(std::declval<Gen&&>().generated_value()))
{
return t.generated_value();
}
static SPROUT_CONSTEXPR decltype(std::declval<Gen const&>().generated_value())
2013-04-22 04:35:01 +00:00
get_generated_value(Gen const& t)
2013-04-22 14:55:30 +00:00
SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(std::declval<Gen const&>().generated_value()))
{
return t.generated_value();
}
};
template<typename Gen>
struct generator_access_traits_generated_value_impl<
Gen,
typename std::enable_if<!sprout::generators::detail::has_mem_generated_value<Gen>::value>::type
> {
public:
static SPROUT_CONSTEXPR decltype(sprout::tuples::get<0>(std::declval<Gen&>()))
2013-04-22 04:35:01 +00:00
get_generated_value(Gen& t)
2013-04-22 14:55:30 +00:00
SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(sprout::tuples::get<0>(std::declval<Gen&>())))
{
return sprout::tuples::get<0>(t);
}
static SPROUT_CONSTEXPR decltype(sprout::tuples::get<0>(std::declval<Gen&&>()))
2013-04-22 04:35:01 +00:00
get_generated_value(Gen&& t)
2013-04-22 14:55:30 +00:00
SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(sprout::tuples::get<0>(std::declval<Gen&>())))
{
return sprout::tuples::get<0>(t);
}
static SPROUT_CONSTEXPR decltype(sprout::tuples::get<0>(std::declval<Gen const&>()))
2013-04-22 04:35:01 +00:00
get_generated_value(Gen const& t)
2013-04-22 14:55:30 +00:00
SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(sprout::tuples::get<0>(std::declval<Gen const&>())))
{
return sprout::tuples::get<0>(t);
}
};
template<typename T>
struct has_mem_next_generator_test {
public:
template<
typename U = T,
typename = decltype(std::declval<U>().next_generator())
>
static sprout::true_type test(int);
static sprout::false_type test(...);
};
#if defined(_MSC_VER)
template<typename T, typename Base_ = decltype(sprout::generators::detail::has_mem_next_generator_test<T>::test(0))>
struct has_mem_next_generator
: public Base_
{};
#else
template<typename T>
struct has_mem_next_generator
: public decltype(sprout::generators::detail::has_mem_next_generator_test<T>::test(0))
{};
#endif
template<typename Gen, typename = void>
struct generator_access_traits_next_generator_impl;
template<typename Gen>
struct generator_access_traits_next_generator_impl<
Gen,
typename std::enable_if<sprout::generators::detail::has_mem_next_generator<Gen>::value>::type
> {
public:
static SPROUT_CONSTEXPR decltype(std::declval<Gen&>().next_generator())
2013-04-22 04:35:01 +00:00
get_next_generator(Gen& t)
2013-04-22 14:55:30 +00:00
SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(std::declval<Gen&>().next_generator()))
{
return t.next_generator();
}
static SPROUT_CONSTEXPR decltype(std::declval<Gen&&>().next_generator())
2013-04-22 04:35:01 +00:00
get_next_generator(Gen&& t)
2013-04-22 14:55:30 +00:00
SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(std::declval<Gen&&>().next_generator()))
{
return t.next_generator();
}
static SPROUT_CONSTEXPR decltype(std::declval<Gen const&>().next_generator())
2013-04-22 04:35:01 +00:00
get_next_generator(Gen const& t)
2013-04-22 14:55:30 +00:00
SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(std::declval<Gen const&>().next_generator()))
{
return t.next_generator();
}
};
template<typename Gen>
struct generator_access_traits_next_generator_impl<
Gen,
typename std::enable_if<!sprout::generators::detail::has_mem_next_generator<Gen>::value>::type
> {
public:
static SPROUT_CONSTEXPR decltype(sprout::tuples::get<1>(std::declval<Gen&>()))
2013-04-22 04:35:01 +00:00
get_next_generator(Gen& t)
2013-04-22 14:55:30 +00:00
SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(sprout::tuples::get<1>(std::declval<Gen&>())))
{
return sprout::tuples::get<1>(t);
}
static SPROUT_CONSTEXPR decltype(sprout::tuples::get<1>(std::declval<Gen&&>()))
2013-04-22 04:35:01 +00:00
get_next_generator(Gen&& t)
2013-04-22 14:55:30 +00:00
SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(sprout::tuples::get<1>(std::declval<Gen&>())))
{
return sprout::tuples::get<1>(t);
}
static SPROUT_CONSTEXPR decltype(sprout::tuples::get<1>(std::declval<Gen const&>()))
2013-04-22 04:35:01 +00:00
get_next_generator(Gen const& t)
2013-04-22 14:55:30 +00:00
SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(sprout::tuples::get<1>(std::declval<Gen const&>())))
{
return sprout::tuples::get<1>(t);
}
};
} // namespace detail
//
// generator_access_traits
//
template<typename Gen>
struct generator_access_traits
: public sprout::generators::detail::generator_access_traits_generated_value_impl<Gen>
, public sprout::generators::detail::generator_access_traits_next_generator_impl<Gen>
{};
template<typename Gen>
struct generator_access_traits<Gen const> {
public:
2013-04-22 04:35:01 +00:00
static SPROUT_CONSTEXPR decltype(sprout::generators::generator_access_traits<Gen>::get_generated_value(std::declval<Gen const&>()))
get_generated_value(Gen const& t)
2013-04-22 14:55:30 +00:00
SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(sprout::generators::generator_access_traits<Gen>::get_generated_value(std::declval<Gen const&>())))
{
2013-04-22 04:35:01 +00:00
return sprout::generators::generator_access_traits<Gen>::get_generated_value(t);
}
2013-04-22 04:35:01 +00:00
static SPROUT_CONSTEXPR decltype(sprout::generators::generator_access_traits<Gen>::get_next_generator(std::declval<Gen const&>()))
get_next_generator(Gen const& t)
2013-04-22 14:55:30 +00:00
SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(sprout::generators::generator_access_traits<Gen>::get_next_generator(std::declval<Gen const&>())))
{
2013-04-22 04:35:01 +00:00
return sprout::generators::generator_access_traits<Gen>::get_next_generator(t);
}
};
} // namespace generators
using sprout::generators::generator_access_traits;
} // namespace sprout
#endif // #ifndef SPROUT_GENERATOR_GENERATOR_ACCESS_TRAITS_HPP