Sprout/sprout/container/container_traits.hpp

470 lines
18 KiB
C++
Raw Normal View History

#ifndef SPROUT_CONTAINER_CONTAINER_TRAITS_HPP
#define SPROUT_CONTAINER_CONTAINER_TRAITS_HPP
#include <cstddef>
#include <iterator>
#include <array>
#include <type_traits>
#include <sprout/config.hpp>
#include <sprout/type_traits/has_xxx.hpp>
#include <sprout/type_traits/inherit_if_xxx.hpp>
namespace sprout {
template<typename Container>
struct container_traits;
namespace detail {
//
// has_value_type
// has_iterator
// has_const_iterator
// has_reference
// has_const_reference
// has_size_type
// has_difference_type
// has_pointer
// has_const_pointer
// has_static_size
//
SPROUT_HAS_XXX_TYPE_DEF_LAZY(value_type);
SPROUT_HAS_XXX_TYPE_DEF_LAZY(iterator);
SPROUT_HAS_XXX_TYPE_DEF_LAZY(const_iterator);
SPROUT_HAS_XXX_TYPE_DEF_LAZY(reference);
SPROUT_HAS_XXX_TYPE_DEF_LAZY(const_reference);
SPROUT_HAS_XXX_TYPE_DEF_LAZY(size_type);
SPROUT_HAS_XXX_TYPE_DEF_LAZY(difference_type);
SPROUT_HAS_XXX_TYPE_DEF_LAZY(pointer);
SPROUT_HAS_XXX_TYPE_DEF_LAZY(const_pointer);
SPROUT_HAS_XXX_VALUE_DEF_LAZY(static_size);
//
// inherit_if_value_type
// inherit_if_iterator
// inherit_if_const_iterator
// inherit_if_reference
// inherit_if_const_reference
// inherit_if_size_type
// inherit_if_difference_type
// inherit_if_pointer
// inherit_if_const_pointer
// inherit_if_static_size
//
SPROUT_INHERIT_IF_XXX_TYPE_DEF_LAZY(value_type);
SPROUT_INHERIT_IF_XXX_TYPE_DEF_LAZY(iterator);
SPROUT_INHERIT_IF_XXX_TYPE_DEF_LAZY(const_iterator);
SPROUT_INHERIT_IF_XXX_TYPE_DEF_LAZY(reference);
SPROUT_INHERIT_IF_XXX_TYPE_DEF_LAZY(const_reference);
SPROUT_INHERIT_IF_XXX_TYPE_DEF_LAZY(size_type);
SPROUT_INHERIT_IF_XXX_TYPE_DEF_LAZY(difference_type);
SPROUT_INHERIT_IF_XXX_TYPE_DEF_LAZY(pointer);
SPROUT_INHERIT_IF_XXX_TYPE_DEF_LAZY(const_pointer);
SPROUT_INHERIT_IF_XXX_CONSTANT_DEF_LAZY(static_size);
//
// inherit_iterator_if_const_iterator
// inherit_const_iterator_if_iterator
// inherit_reference_if_const_reference
// inherit_const_reference_if_reference
// inherit_pointer_if_const_pointer
// inherit_const_pointer_if_pointer
//
SPROUT_INHERIT_ALIAS_IF_XXX_TYPE_DEF_LAZY(iterator, const_iterator);
SPROUT_INHERIT_ALIAS_IF_XXX_TYPE_DEF_LAZY(const_iterator, iterator);
SPROUT_INHERIT_ALIAS_IF_XXX_TYPE_DEF_LAZY(reference, const_reference);
SPROUT_INHERIT_ALIAS_IF_XXX_TYPE_DEF_LAZY(const_reference, reference);
SPROUT_INHERIT_ALIAS_IF_XXX_TYPE_DEF_LAZY(pointer, const_pointer);
SPROUT_INHERIT_ALIAS_IF_XXX_TYPE_DEF_LAZY(const_pointer, pointer);
//
// has_container_nosy_iterator
//
template<typename Container>
struct has_container_nosy_iterator
: public std::integral_constant<
bool,
sprout::detail::has_iterator<Container>::value
|| sprout::detail::has_const_iterator<Container>::value
>
{};
//
// has_container_nosy_value_type
//
template<typename Container>
struct has_container_nosy_value_type
: public std::integral_constant<
bool,
sprout::detail::has_value_type<Container>::value
|| sprout::detail::has_container_nosy_iterator<Container>::value
>
{};
//
// container_nosy_iterator
//
template<typename Container, bool HasIterator, bool HasConstIterator>
struct container_nosy_iterator_impl {};
template<typename Container, bool HasConstIterator>
struct container_nosy_iterator_impl<Container, true, HasConstIterator>
: public sprout::detail::inherit_if_iterator<Container>
{};
template<typename Container>
struct container_nosy_iterator_impl<Container, false, true>
: public sprout::detail::inherit_iterator_if_const_iterator<Container>
{};
template<typename Container>
struct container_nosy_iterator
: public sprout::detail::container_nosy_iterator_impl<
Container,
sprout::detail::has_iterator<Container>::value,
sprout::detail::has_const_iterator<Container>::value
>
{};
//
// container_nosy_const_iterator
//
template<typename Container, bool HasConstIterator, bool HasIterator>
struct container_nosy_const_iterator_impl {};
template<typename Container, bool HasIterator>
struct container_nosy_const_iterator_impl<Container, true, HasIterator>
: public sprout::detail::inherit_if_const_iterator<Container>
{};
template<typename Container>
struct container_nosy_const_iterator_impl<Container, false, true>
: public sprout::detail::inherit_const_iterator_if_iterator<Container>
{};
template<typename Container>
struct container_nosy_const_iterator
: public sprout::detail::container_nosy_const_iterator_impl<
Container,
sprout::detail::has_const_iterator<Container>::value,
sprout::detail::has_iterator<Container>::value
>
{};
//
// container_nosy_value_type
//
template<typename Container, bool HasValueType, bool HasNosyIterator>
struct container_nosy_value_type_impl {};
template<typename Container, bool HasNosyIterator>
struct container_nosy_value_type_impl<Container, true, HasNosyIterator> {
public:
typedef typename Container::value_type value_type;
};
template<typename Container>
struct container_nosy_value_type_impl<Container, false, true> {
public:
typedef typename std::iterator_traits<
typename sprout::detail::container_nosy_iterator<Container>::iterator
>::value_type value_type;
};
template<typename Container>
struct container_nosy_value_type
: public sprout::detail::container_nosy_value_type_impl<
Container,
sprout::detail::has_value_type<Container>::value,
sprout::detail::has_container_nosy_iterator<Container>::value
>
{};
//
// container_nosy_reference
//
template<typename Container, bool HasReference, bool HasConstReference, bool HasNosyIterator>
struct container_nosy_reference_impl {};
template<typename Container, bool HasConstReference, bool HasNosyIterator>
struct container_nosy_reference_impl<Container, true, HasConstReference, HasNosyIterator> {
public:
typedef typename Container::reference reference;
};
template<typename Container, bool HasNosyIterator>
struct container_nosy_reference_impl<Container, false, true, HasNosyIterator> {
public:
typedef typename Container::const_reference reference;
};
template<typename Container>
struct container_nosy_reference_impl<Container, false, false, true> {
public:
typedef typename std::iterator_traits<
typename sprout::detail::container_nosy_iterator<Container>::iterator
>::reference reference;
};
template<typename Container>
struct container_nosy_reference
: public sprout::detail::container_nosy_reference_impl<
Container,
sprout::detail::has_reference<Container>::value,
sprout::detail::has_const_reference<Container>::value,
sprout::detail::has_container_nosy_iterator<Container>::value
>
{};
//
// container_nosy_const_reference
//
template<typename Container, bool HasConstReference, bool HasReference, bool HasNosyIterator>
struct container_nosy_const_reference_impl {};
template<typename Container, bool HasReference, bool HasNosyIterator>
struct container_nosy_const_reference_impl<Container, true, HasReference, HasNosyIterator> {
public:
typedef typename Container::const_reference const_reference;
};
template<typename Container, bool HasNosyIterator>
struct container_nosy_const_reference_impl<Container, false, true, HasNosyIterator> {
public:
typedef typename Container::reference const_reference;
};
template<typename Container>
struct container_nosy_const_reference_impl<Container, false, false, true> {
public:
typedef typename std::iterator_traits<
typename sprout::detail::container_nosy_const_iterator<Container>::const_iterator
>::reference const_reference;
};
template<typename Container>
struct container_nosy_const_reference
: public sprout::detail::container_nosy_const_reference_impl<
Container,
sprout::detail::has_const_reference<Container>::value,
sprout::detail::has_reference<Container>::value,
sprout::detail::has_container_nosy_iterator<Container>::value
>
{};
//
// container_nosy_difference_type
//
template<typename Container, bool HasDifferenceType, bool HasNosyIterator>
struct container_nosy_difference_type_impl {
public:
typedef std::ptrdiff_t difference_type;
};
template<typename Container, bool HasNosyIterator>
struct container_nosy_difference_type_impl<Container, true, HasNosyIterator> {
public:
typedef typename Container::difference_type difference_type;
};
template<typename Container>
struct container_nosy_difference_type_impl<Container, false, true> {
public:
typedef typename std::iterator_traits<
typename sprout::detail::container_nosy_iterator<Container>::iterator
>::difference_type difference_type;
};
template<typename Container>
struct container_nosy_difference_type
: public sprout::detail::container_nosy_difference_type_impl<
Container,
sprout::detail::has_difference_type<Container>::value,
sprout::detail::has_container_nosy_iterator<Container>::value
>
{};
//
// container_nosy_size_type
//
template<typename Container, bool HasSizeType, bool HasNosyIterator>
struct container_nosy_size_type_impl {
public:
typedef std::size_t size_type;
};
template<typename Container, bool HasNosyIterator>
struct container_nosy_size_type_impl<Container, true, HasNosyIterator> {
public:
typedef typename Container::size_type size_type;
};
template<typename Container>
struct container_nosy_size_type_impl<Container, false, true> {
public:
typedef typename std::make_unsigned<
typename std::iterator_traits<
typename sprout::detail::container_nosy_iterator<Container>::iterator
>::difference_type
>::type size_type;
};
template<typename Container>
struct container_nosy_size_type
: public sprout::detail::container_nosy_size_type_impl<
Container,
sprout::detail::has_size_type<Container>::value,
sprout::detail::has_container_nosy_iterator<Container>::value
>
{};
//
// container_nosy_pointer
//
template<typename Container, bool HasPointer, bool HasConstPointer, bool HasNosyValueType>
struct container_nosy_pointer_impl {};
template<typename Container, bool HasConstPointer, bool HasNosyValueType>
struct container_nosy_pointer_impl<Container, true, HasConstPointer, HasNosyValueType> {
public:
typedef typename Container::pointer pointer;
};
template<typename Container, bool HasNosyValueType>
struct container_nosy_pointer_impl<Container, false, true, HasNosyValueType> {
public:
typedef typename Container::const_pointer pointer;
};
template<typename Container>
struct container_nosy_pointer_impl<Container, false, false, true> {
public:
typedef typename sprout::detail::container_nosy_value_type<Container>::value_type* pointer;
};
template<typename Container>
struct container_nosy_pointer
: public sprout::detail::container_nosy_pointer_impl<
Container,
sprout::detail::has_pointer<Container>::value,
sprout::detail::has_const_pointer<Container>::value,
sprout::detail::has_container_nosy_value_type<Container>::value
>
{};
//
// container_nosy_const_pointer
//
template<typename Container, bool HasConstPointer, bool HasPointer, bool HasNosyValueType>
struct container_nosy_const_pointer_impl {};
template<typename Container, bool HasPointer, bool HasNosyValueType>
struct container_nosy_const_pointer_impl<Container, true, HasPointer, HasNosyValueType> {
public:
typedef typename Container::const_pointer const_pointer;
};
template<typename Container, bool HasNosyValueType>
struct container_nosy_const_pointer_impl<Container, false, true, HasNosyValueType> {
public:
typedef typename Container::pointer const_pointer;
};
template<typename Container>
struct container_nosy_const_pointer_impl<Container, false, false, true> {
public:
typedef typename sprout::detail::container_nosy_value_type<Container>::value_type const* const_pointer;
};
template<typename Container>
struct container_nosy_const_pointer
: public sprout::detail::container_nosy_const_pointer_impl<
Container,
sprout::detail::has_const_pointer<Container>::value,
sprout::detail::has_pointer<Container>::value,
sprout::detail::has_container_nosy_value_type<Container>::value
>
{};
//
// container_nosy_static_size
//
template<typename Container>
struct container_nosy_static_size
: public sprout::detail::inherit_if_static_size<Container>
{};
//
// container_nosy_fixed_size
//
template<typename Container, bool HasStaticSize>
struct container_nosy_fixed_size_impl {};
template<typename Container>
struct container_nosy_fixed_size_impl<Container, true> {
public:
static SPROUT_CONSTEXPR decltype(Container::static_size) fixed_size() {
return Container::static_size;
}
};
template<typename Container>
struct container_nosy_fixed_size
: public sprout::detail::container_nosy_fixed_size_impl<
Container,
sprout::detail::has_static_size<Container>::value
>
{};
//
// container_traits_default
//
template<typename Container>
struct container_traits_default
: public sprout::detail::container_nosy_value_type<Container>
, public sprout::detail::container_nosy_iterator<Container>
, public sprout::detail::container_nosy_const_iterator<Container>
, public sprout::detail::container_nosy_reference<Container>
, public sprout::detail::container_nosy_const_reference<Container>
, public sprout::detail::container_nosy_size_type<Container>
, public sprout::detail::container_nosy_difference_type<Container>
, public sprout::detail::container_nosy_pointer<Container>
, public sprout::detail::container_nosy_const_pointer<Container>
, public sprout::detail::container_nosy_static_size<Container>
, public sprout::detail::container_nosy_fixed_size<Container>
{};
template<typename T, std::size_t N>
struct container_traits_default<T[N]> {
public:
typedef T value_type;
typedef T* iterator;
typedef T const* const_iterator;
typedef T& reference;
typedef T const& const_reference;
typedef std::size_t size_type;
typedef std::ptrdiff_t difference_type;
typedef T* pointer;
typedef T const* const_pointer;
public:
SPROUT_STATIC_CONSTEXPR size_type static_size = N ;
public:
static SPROUT_CONSTEXPR size_type fixed_size() {
return static_size;
}
};
//
// container_traits_const_default
//
template<typename Container>
struct container_traits_const_default
: public sprout::detail::inherit_if_value_type<sprout::container_traits<Container> >
, public sprout::detail::inherit_if_const_iterator<sprout::container_traits<Container> >
, public sprout::detail::inherit_if_const_reference<sprout::container_traits<Container> >
, public sprout::detail::inherit_if_size_type<sprout::container_traits<Container> >
, public sprout::detail::inherit_if_difference_type<sprout::container_traits<Container> >
, public sprout::detail::inherit_if_const_pointer<sprout::container_traits<Container> >
, public sprout::detail::inherit_if_static_size<sprout::container_traits<Container> >
, public sprout::detail::container_nosy_fixed_size<sprout::container_traits<Container> >
, public sprout::detail::inherit_iterator_if_const_iterator<sprout::container_traits<Container> >
, public sprout::detail::inherit_reference_if_const_reference<sprout::container_traits<Container> >
, public sprout::detail::inherit_pointer_if_const_pointer<sprout::container_traits<Container> >
{};
} // namespace detail
//
// container_traits
//
template<typename Container>
struct container_traits
: public sprout::detail::container_traits_default<Container>
{};
template<typename Container>
struct container_traits<Container const>
: public sprout::detail::container_traits_const_default<Container>
{};
template<typename T, std::size_t N>
struct container_traits<T[N]>
: public sprout::detail::container_traits_default<T[N]>
{};
template<typename T, std::size_t N>
struct container_traits<T const[N]>
: public sprout::detail::container_traits_const_default<T[N]>
{};
//
// container_traits_facade
//
template<typename Container>
struct container_traits_facade
: public sprout::detail::inherit_if_value_type<sprout::container_traits<Container> >
, public sprout::detail::inherit_if_iterator<sprout::container_traits<Container> >
, public sprout::detail::inherit_if_const_iterator<sprout::container_traits<Container> >
, public sprout::detail::inherit_if_reference<sprout::container_traits<Container> >
, public sprout::detail::inherit_if_const_reference<sprout::container_traits<Container> >
, public sprout::detail::inherit_if_size_type<sprout::container_traits<Container> >
, public sprout::detail::inherit_if_difference_type<sprout::container_traits<Container> >
, public sprout::detail::inherit_if_pointer<sprout::container_traits<Container> >
, public sprout::detail::inherit_if_const_pointer<sprout::container_traits<Container> >
, public sprout::detail::inherit_if_static_size<sprout::container_traits<Container> >
, public sprout::detail::container_nosy_fixed_size<sprout::container_traits<Container> >
{};
} // namespace sprout
#endif // #ifndef SPROUT_CONTAINER_CONTAINER_TRAITS_HPP