174 lines
5.9 KiB
Plaintext
174 lines
5.9 KiB
Plaintext
|
/*=============================================================================
|
||
|
Copyright (c) 2004 Angus Leeming
|
||
|
Copyright (c) 2004 Joel de Guzman
|
||
|
|
||
|
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 BOOST_PHOENIX_CONTAINER_DETAIL_CONTAINER_HPP
|
||
|
#define BOOST_PHOENIX_CONTAINER_DETAIL_CONTAINER_HPP
|
||
|
|
||
|
#include <utility>
|
||
|
#include <boost/mpl/eval_if.hpp>
|
||
|
#include <boost/type_traits/is_same.hpp>
|
||
|
#include <boost/type_traits/is_const.hpp>
|
||
|
|
||
|
namespace boost { namespace phoenix { namespace stl
|
||
|
{
|
||
|
///////////////////////////////////////////////////////////////////////////////
|
||
|
//
|
||
|
// Metafunctions "value_type_of", "key_type_of" etc.
|
||
|
//
|
||
|
// These metafunctions define a typedef "type" that returns the nested
|
||
|
// type if it exists. If not then the typedef returns void.
|
||
|
//
|
||
|
// For example, "value_type_of<std::vector<int> >::type" is "int" whilst
|
||
|
// "value_type_of<double>::type" is "void".
|
||
|
//
|
||
|
// I use a macro to define structs "value_type_of" etc simply to cut
|
||
|
// down on the amount of code. The macro is #undef-ed immediately after
|
||
|
// its final use.
|
||
|
//
|
||
|
/////////////////////////////////////////////////////////////////c//////////////
|
||
|
#define MEMBER_TYPE_OF(MEMBER_TYPE) \
|
||
|
template <typename C> \
|
||
|
struct BOOST_PP_CAT(MEMBER_TYPE, _of) \
|
||
|
{ \
|
||
|
typedef typename C::MEMBER_TYPE type; \
|
||
|
}
|
||
|
|
||
|
MEMBER_TYPE_OF(allocator_type);
|
||
|
MEMBER_TYPE_OF(const_iterator);
|
||
|
MEMBER_TYPE_OF(const_reference);
|
||
|
MEMBER_TYPE_OF(const_reverse_iterator);
|
||
|
MEMBER_TYPE_OF(container_type);
|
||
|
MEMBER_TYPE_OF(data_type);
|
||
|
MEMBER_TYPE_OF(iterator);
|
||
|
MEMBER_TYPE_OF(key_compare);
|
||
|
MEMBER_TYPE_OF(key_type);
|
||
|
MEMBER_TYPE_OF(reference);
|
||
|
MEMBER_TYPE_OF(reverse_iterator);
|
||
|
MEMBER_TYPE_OF(size_type);
|
||
|
MEMBER_TYPE_OF(value_compare);
|
||
|
MEMBER_TYPE_OF(value_type);
|
||
|
|
||
|
#undef MEMBER_TYPE_OF
|
||
|
|
||
|
///////////////////////////////////////////////////////////////////////////////
|
||
|
//
|
||
|
// Const-Qualified types.
|
||
|
//
|
||
|
// Many of the stl member functions have const and non-const
|
||
|
// overloaded versions that return distinct types. For example:
|
||
|
//
|
||
|
// iterator begin();
|
||
|
// const_iterator begin() const;
|
||
|
//
|
||
|
// The three class templates defined below,
|
||
|
// const_qualified_reference_of, const_qualified_iterator_of
|
||
|
// and const_qualified_reverse_iterator_of provide a means to extract
|
||
|
// this return type automatically.
|
||
|
//
|
||
|
///////////////////////////////////////////////////////////////////////////////
|
||
|
template <typename C>
|
||
|
struct const_qualified_reference_of
|
||
|
{
|
||
|
typedef typename
|
||
|
boost::mpl::eval_if_c<
|
||
|
boost::is_const<C>::value
|
||
|
, const_reference_of<C>
|
||
|
, reference_of<C>
|
||
|
>::type
|
||
|
type;
|
||
|
};
|
||
|
|
||
|
template <typename C>
|
||
|
struct const_qualified_iterator_of
|
||
|
{
|
||
|
typedef typename
|
||
|
boost::mpl::eval_if_c<
|
||
|
boost::is_const<C>::value
|
||
|
, const_iterator_of<C>
|
||
|
, iterator_of<C>
|
||
|
>::type
|
||
|
type;
|
||
|
};
|
||
|
|
||
|
template <typename C>
|
||
|
struct const_qualified_reverse_iterator_of
|
||
|
{
|
||
|
typedef typename
|
||
|
boost::mpl::eval_if_c<
|
||
|
boost::is_const<C>::value
|
||
|
, const_reverse_iterator_of<C>
|
||
|
, reverse_iterator_of<C>
|
||
|
>::type
|
||
|
type;
|
||
|
};
|
||
|
|
||
|
///////////////////////////////////////////////////////////////////////////////
|
||
|
//
|
||
|
// has_mapped_type<C>
|
||
|
//
|
||
|
// Given a container C, determine if it is a map or multimap
|
||
|
// by checking if it has a member type named "mapped_type".
|
||
|
//
|
||
|
///////////////////////////////////////////////////////////////////////////////
|
||
|
namespace stl_impl
|
||
|
{
|
||
|
struct one { char a[1]; };
|
||
|
struct two { char a[2]; };
|
||
|
|
||
|
template <typename C>
|
||
|
one has_mapped_type(typename C::mapped_type(*)());
|
||
|
|
||
|
template <typename C>
|
||
|
two has_mapped_type(...);
|
||
|
}
|
||
|
|
||
|
template <typename C>
|
||
|
struct has_mapped_type
|
||
|
: boost::mpl::bool_<
|
||
|
sizeof(stl_impl::has_mapped_type<C>(0)) == sizeof(stl_impl::one)
|
||
|
>
|
||
|
{};
|
||
|
|
||
|
///////////////////////////////////////////////////////////////////////////////
|
||
|
//
|
||
|
// map_insert_returns_pair<C>
|
||
|
//
|
||
|
// Distinguish a map from a multimap by checking the return type
|
||
|
// of its "insert" member function. A map returns a pair while
|
||
|
// a multimap returns an iterator.
|
||
|
//
|
||
|
///////////////////////////////////////////////////////////////////////////////
|
||
|
namespace stl_impl
|
||
|
{
|
||
|
// Cool implementation of map_insert_returns_pair by Daniel Wallin.
|
||
|
// Thanks Daniel!!! I owe you a Pizza!
|
||
|
|
||
|
template<class A, class B>
|
||
|
one map_insert_returns_pair_check(std::pair<A,B> const&);
|
||
|
|
||
|
template <typename T>
|
||
|
two map_insert_returns_pair_check(T const&);
|
||
|
|
||
|
template <typename C>
|
||
|
struct map_insert_returns_pair
|
||
|
{
|
||
|
static typename C::value_type const& get;
|
||
|
BOOST_STATIC_CONSTANT(int,
|
||
|
value = sizeof(
|
||
|
map_insert_returns_pair_check(((C*)0)->insert(get))));
|
||
|
typedef boost::mpl::bool_<value == sizeof(one)> type;
|
||
|
};
|
||
|
}
|
||
|
|
||
|
template <typename C>
|
||
|
struct map_insert_returns_pair
|
||
|
: stl_impl::map_insert_returns_pair<C>::type {};
|
||
|
|
||
|
}}} // namespace boost::phoenix::stl
|
||
|
|
||
|
#endif // BOOST_PHOENIX_STL_CONTAINER_TRAITS_HPP
|