/*============================================================================= Copyright (c) 2011-2014 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_CONTAINER_CONTAINER_RANGE_TRAITS_HPP #define SPROUT_CONTAINER_CONTAINER_RANGE_TRAITS_HPP #include #include #include #include #include #include #include #include #include #include #include #include namespace sprout { // // container_range_traits // namespace detail { template struct has_mem_size_test { public: template< typename U = T, typename = typename sprout::identity().size())>::type > static sprout::true_type test(int); static sprout::false_type test(...); }; #if defined(_MSC_VER) template::test(0))>::type> struct has_mem_size : public Base_ {}; #else template struct has_mem_size : public sprout::identity::test(0))>::type {}; #endif template struct container_range_traits_range_size_impl; template struct container_range_traits_range_size_impl< Container, typename std::enable_if::value>::type > { public: static SPROUT_CONSTEXPR typename sprout::container_traits::size_type range_size(Container const& cont) { return cont.size(); } }; template struct container_range_traits_range_size_impl< Container, typename std::enable_if::value>::type > { public: static SPROUT_CONSTEXPR typename sprout::container_traits::size_type range_size(Container const& cont) { return sprout::distance(sprout::begin(cont), sprout::end(cont)); } }; template struct has_mem_empty_test { public: template< typename U = T, typename = typename sprout::identity().empty())>::type > static sprout::true_type test(int); static sprout::false_type test(...); }; #if defined(_MSC_VER) template::test(0))>::type> struct has_mem_empty : public Base_ {}; #else template struct has_mem_empty : public sprout::identity::test(0))>::type {}; #endif template struct container_range_traits_range_empty_impl; template struct container_range_traits_range_empty_impl< Container, typename std::enable_if::value>::type > { public: static SPROUT_CONSTEXPR bool range_empty(Container const& cont) { return cont.empty(); } }; template struct container_range_traits_range_empty_impl< Container, typename std::enable_if::value>::type > { public: static SPROUT_CONSTEXPR bool range_empty(Container const& cont) { return sprout::size(cont) == 0; } }; } // namespace detail template struct container_range_traits : public sprout::detail::container_range_traits_range_size_impl , public sprout::detail::container_range_traits_range_empty_impl { public: // iterators: static SPROUT_CONSTEXPR typename sprout::container_traits::iterator range_begin(Container& cont) { return cont.begin(); } static SPROUT_CONSTEXPR typename sprout::container_traits::iterator range_begin(Container const& cont) { return cont.begin(); } static SPROUT_CONSTEXPR typename sprout::container_traits::iterator range_end(Container& cont) { return cont.end(); } static SPROUT_CONSTEXPR typename sprout::container_traits::iterator range_end(Container const& cont) { return cont.end(); } }; template struct container_range_traits { public: // iterators: static SPROUT_CONSTEXPR typename sprout::container_traits::iterator range_begin(Container const& cont) { return sprout::container_range_traits::range_begin(cont); } static SPROUT_CONSTEXPR typename sprout::container_traits::iterator range_end(Container const& cont) { return sprout::container_range_traits::range_end(cont); } // capacity: static SPROUT_CONSTEXPR typename sprout::container_traits::size_type range_size(Container const& cont) { return sprout::container_range_traits::range_size(cont); } static SPROUT_CONSTEXPR bool range_empty(Container const& cont) { return sprout::container_range_traits::range_empty(cont); } }; template struct container_range_traits { public: // iterators: static SPROUT_CONSTEXPR typename sprout::container_traits::iterator range_begin(T (& arr)[N]) { typedef typename sprout::container_traits::iterator type; return type(arr); } static SPROUT_CONSTEXPR typename sprout::container_traits::iterator range_begin(T const (& arr)[N]) { typedef typename sprout::container_traits::iterator type; return type(arr); } static SPROUT_CONSTEXPR typename sprout::container_traits::iterator range_end(T (& arr)[N]) { typedef typename sprout::container_traits::iterator type; return type(arr) + N; } static SPROUT_CONSTEXPR typename sprout::container_traits::iterator range_end(T const (& arr)[N]) { typedef typename sprout::container_traits::iterator type; return type(arr) + N; } // capacity: static SPROUT_CONSTEXPR typename sprout::container_traits::size_type range_size(T const (&)[N]) { return N; } static SPROUT_CONSTEXPR bool range_empty(T const (&)[N]) { return false; } }; template struct container_range_traits { public: // iterators: static SPROUT_CONSTEXPR typename sprout::container_traits::iterator range_begin(T const (& arr)[N]) { return sprout::container_range_traits::range_begin(arr); } static SPROUT_CONSTEXPR typename sprout::container_traits::const_iterator range_end(T const (& arr)[N]) { return sprout::container_range_traits::range_end(arr); } // capacity: static SPROUT_CONSTEXPR typename sprout::container_traits::size_type range_size(T const (& arr)[N]) { return sprout::container_range_traits::range_size(arr); } static SPROUT_CONSTEXPR bool range_empty(T const (& arr)[N]) { return sprout::container_range_traits::range_empty(arr); } }; } // namespace sprout #endif // #ifndef SPROUT_CONTAINER_CONTAINER_RANGE_TRAITS_HPP