/*============================================================================= 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; } }; template struct has_mem_front_test { public: template< typename U = T, typename = typename sprout::identity().front())>::type > static sprout::true_type test(int); static sprout::false_type test(...); }; #if defined(_MSC_VER) template::test(0))>::type> struct has_mem_front : public Base_ {}; #else template struct has_mem_front : public sprout::identity::test(0))>::type {}; #endif template struct container_range_traits_range_front_impl; template struct container_range_traits_range_front_impl< Container, typename std::enable_if::value>::type > { public: static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_front(Container& cont) { return cont.front(); } }; template struct container_range_traits_range_front_impl< Container, typename std::enable_if::value>::type > { public: static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_front(Container& cont) { return *sprout::begin(cont); } }; template struct container_range_traits_range_front_const_impl; template struct container_range_traits_range_front_const_impl< Container, typename std::enable_if::value>::type > { public: static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_front(Container const& cont) { return cont.front(); } }; template struct container_range_traits_range_front_const_impl< Container, typename std::enable_if::value>::type > { public: static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_front(Container const& cont) { return *sprout::begin(cont); } }; template struct has_mem_back_test { public: template< typename U = T, typename = typename sprout::identity().back())>::type > static sprout::true_type test(int); static sprout::false_type test(...); }; #if defined(_MSC_VER) template::test(0))>::type> struct has_mem_back : public Base_ {}; #else template struct has_mem_back : public sprout::identity::test(0))>::type {}; #endif template struct container_range_traits_range_back_impl; template struct container_range_traits_range_back_impl< Container, typename std::enable_if::value>::type > { public: static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_back(Container& cont) { return cont.back(); } }; template struct container_range_traits_range_back_impl< Container, typename std::enable_if::value>::type > { public: static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_back(Container& cont) { return *sprout::prev(sprout::end(cont)); } }; template struct container_range_traits_range_back_const_impl; template struct container_range_traits_range_back_const_impl< Container, typename std::enable_if::value>::type > { public: static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_back(Container const& cont) { return cont.back(); } }; template struct container_range_traits_range_back_const_impl< Container, typename std::enable_if::value>::type > { public: static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_back(Container const& cont) { return *sprout::prev(sprout::end(cont)); } }; template struct has_mem_at_test { public: template< typename U = T, typename = typename sprout::identity().at(std::declval::size_type>()))>::type > static sprout::true_type test(int); static sprout::false_type test(...); }; #if defined(_MSC_VER) template::test(0))>::type> struct has_mem_at : public Base_ {}; #else template struct has_mem_at : public sprout::identity::test(0))>::type {}; #endif template struct container_range_traits_range_at_impl; template struct container_range_traits_range_at_impl< Container, typename std::enable_if::value>::type > { public: static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_at(Container& cont, typename sprout::container_traits::size_type i) { return cont.at(i); } }; template struct container_range_traits_range_at_impl< Container, typename std::enable_if::value>::type > { public: static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_at(Container& cont, typename sprout::container_traits::size_type i) { return *sprout::next(sprout::begin(cont), i); } }; template struct container_range_traits_range_at_const_impl; template struct container_range_traits_range_at_const_impl< Container, typename std::enable_if::value>::type > { public: static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_at(Container const& cont, typename sprout::container_traits::size_type i) { return cont.at(i); } }; template struct container_range_traits_range_at_const_impl< Container, typename std::enable_if::value>::type > { public: static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_at(Container const& cont, typename sprout::container_traits::size_type i) { return *sprout::next(sprout::begin(cont), i); } }; // // container_range_traits_default // template struct container_range_traits_default : public sprout::detail::container_range_traits_range_size_impl , public sprout::detail::container_range_traits_range_empty_impl , public sprout::detail::container_range_traits_range_front_impl , public sprout::detail::container_range_traits_range_front_const_impl , public sprout::detail::container_range_traits_range_back_impl , public sprout::detail::container_range_traits_range_back_const_impl , public sprout::detail::container_range_traits_range_at_impl , public sprout::detail::container_range_traits_range_at_const_impl { using sprout::detail::container_range_traits_range_front_impl::range_front; using sprout::detail::container_range_traits_range_front_const_impl::range_front; using sprout::detail::container_range_traits_range_back_impl::range_back; using sprout::detail::container_range_traits_range_back_const_impl::range_back; using sprout::detail::container_range_traits_range_at_impl::range_at; using sprout::detail::container_range_traits_range_at_const_impl::range_at; 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(); } // data access: static SPROUT_CONSTEXPR typename sprout::container_traits::pointer range_data(Container& cont) { return cont.data(); } static SPROUT_CONSTEXPR typename sprout::container_traits::pointer range_data(Container const& cont) { return cont.data(); } }; // // container_range_traits_const_default // template struct container_range_traits_const_default { 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); } // element access: static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_front(Container const& cont) { return sprout::container_range_traits::range_front(cont); } static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_back(Container const& cont) { return sprout::container_range_traits::range_back(cont); } static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_at(Container const& cont, typename sprout::container_traits::size_type i) { return sprout::container_range_traits::range_at(cont, i); } // data access: static SPROUT_CONSTEXPR typename sprout::container_traits::pointer range_data(Container const& cont) { return sprout::container_range_traits::range_data(cont); } }; } // namespace detail // // container_range_traits // template struct container_range_traits : public sprout::detail::container_range_traits_default {}; template struct container_range_traits : public sprout::detail::container_range_traits_const_default {}; 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; } // element access: static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_front(T (& arr)[N]) { return arr[0]; } static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_front(T const (& arr)[N]) { return arr[0]; } static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_back(T (& arr)[N]) { return arr[N - 1]; } static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_back(T const (& arr)[N]) { return arr[N - 1]; } static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_at(T (& arr)[N], typename sprout::container_traits::size_type i) { return arr[i]; } static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_at(T const (& arr)[N], typename sprout::container_traits::size_type i) { return arr[i]; } // data access: static SPROUT_CONSTEXPR typename sprout::container_traits::pointer range_data(T (& arr)[N]) { return &arr[0]; } static SPROUT_CONSTEXPR typename sprout::container_traits::pointer range_data(T const (& arr)[N]) { return &arr[0]; } }; 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::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); } // element access: static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_front(T const (& arr)[N]) { return sprout::container_range_traits::range_front(arr); } static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_back(T const (& arr)[N]) { return sprout::container_range_traits::range_back(arr); } static SPROUT_CONSTEXPR typename sprout::container_traits::reference range_at(T const (& arr)[N], typename sprout::container_traits::size_type i) { return sprout::container_range_traits::range_at(arr, i); } // data access: static SPROUT_CONSTEXPR typename sprout::container_traits::pointer range_data(T const (& arr)[N]) { return sprout::container_range_traits::range_data(arr); } }; } // namespace sprout #endif // #ifndef SPROUT_CONTAINER_CONTAINER_RANGE_TRAITS_HPP