Sprout/sprout/string/char_traits.hpp

267 lines
9.5 KiB
C++
Raw Normal View History

2013-08-08 09:54:33 +00:00
/*=============================================================================
Copyright (c) 2011-2019 Bolero MURAKAMI
2013-08-08 09:54:33 +00:00
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)
=============================================================================*/
2012-04-14 10:06:21 +00:00
#ifndef SPROUT_STRING_CHAR_TRAITS_HPP
#define SPROUT_STRING_CHAR_TRAITS_HPP
#include <cstdio>
2012-04-14 10:06:21 +00:00
#include <string>
#include <sprout/config.hpp>
#include <sprout/workaround/std/cstddef.hpp>
#include <sprout/functional/bind2nd.hpp>
2016-03-23 09:18:26 +00:00
#include <sprout/iterator/next.hpp>
#include <sprout/iterator/ptr_index_iterator.hpp>
2013-01-11 19:08:44 +00:00
#include <sprout/algorithm/find_if.hpp>
2012-12-21 14:12:54 +00:00
#include <sprout/algorithm/tristate_lexicographical_compare.hpp>
#include <sprout/algorithm/cxx14/copy.hpp>
#include <sprout/algorithm/cxx14/copy_backward.hpp>
#include <sprout/algorithm/cxx14/fill.hpp>
#include <sprout/iterator/operation.hpp>
2013-01-12 16:16:48 +00:00
#include <sprout/cstring/strlen.hpp>
2012-04-14 10:06:21 +00:00
namespace sprout {
2013-02-19 16:12:56 +00:00
namespace detail {
template<typename Traits>
class char_traits_eq {
public:
typedef Traits traits_type;
typedef typename traits_type::char_type char_type;
typedef bool result_type;
typedef char_type first_argument_type;
typedef char_type second_argument_type;
public:
SPROUT_CONSTEXPR bool operator()(char_type c1, char_type c2) const SPROUT_NOEXCEPT {
return traits_type::eq(c1, c2);
}
};
template<typename Traits>
class char_traits_lt {
public:
typedef Traits traits_type;
typedef typename traits_type::char_type char_type;
typedef bool result_type;
typedef char_type first_argument_type;
typedef char_type second_argument_type;
public:
SPROUT_CONSTEXPR bool operator()(char_type c1, char_type c2) const SPROUT_NOEXCEPT {
return traits_type::lt(c1, c2);
}
};
} // namespace detail
2012-04-14 10:06:21 +00:00
//
// char_traits
//
template<typename Char>
class char_traits {
private:
typedef std::char_traits<Char> impl_type;
public:
typedef typename impl_type::char_type char_type;
typedef typename impl_type::int_type int_type;
typedef typename impl_type::off_type off_type;
typedef typename impl_type::pos_type pos_type;
typedef typename impl_type::state_type state_type;
private:
static SPROUT_CONSTEXPR char_type const* find_impl(char_type const* found, char_type const* last) {
return found == last ? nullptr
: found
;
}
2012-04-14 10:06:21 +00:00
public:
static SPROUT_CXX14_CONSTEXPR void assign(char_type& c1, char_type const& c2) SPROUT_NOEXCEPT {
c1 = c2;
2012-04-14 10:06:21 +00:00
}
#ifdef SPROUT_NO_CXX11_CHAR_TRAITS
static SPROUT_CONSTEXPR bool eq(char_type c1, char_type c2) SPROUT_NOEXCEPT {
return c1 == c2;
}
static SPROUT_CONSTEXPR bool lt(char_type c1, char_type c2) SPROUT_NOEXCEPT {
return c1 < c2;
}
#else
2012-04-14 10:06:21 +00:00
static SPROUT_CONSTEXPR bool eq(char_type c1, char_type c2) SPROUT_NOEXCEPT {
return impl_type::eq(c1, c2);
}
static SPROUT_CONSTEXPR bool lt(char_type c1, char_type c2) SPROUT_NOEXCEPT {
return impl_type::lt(c1, c2);
}
#endif
2012-04-14 10:06:21 +00:00
static SPROUT_CONSTEXPR int compare(char_type const* s1, char_type const* s2, std::size_t n) {
2012-12-21 14:12:54 +00:00
return sprout::tristate_lexicographical_compare(
2013-02-26 08:03:30 +00:00
sprout::ptr_index(s1), sprout::ptr_index(s1, n), char_type(),
sprout::ptr_index(s2), sprout::ptr_index(s2, n), char_type(),
2013-02-19 16:12:56 +00:00
sprout::detail::char_traits_lt<char_traits>()
2012-12-21 14:12:54 +00:00
);
2012-04-14 10:06:21 +00:00
}
static SPROUT_CONSTEXPR std::size_t length(char_type const* s) {
2013-02-19 16:12:56 +00:00
return sprout::strlen(s);
2012-04-14 10:06:21 +00:00
}
static SPROUT_CONSTEXPR char_type const* find(char_type const* s, std::size_t n, char_type const& a) {
return find_impl(
2013-02-26 08:03:30 +00:00
sprout::ptr_unindex(
2013-01-11 19:08:44 +00:00
sprout::find_if(
2013-02-26 08:03:30 +00:00
sprout::ptr_index(s), sprout::ptr_index(s, n),
2013-02-19 16:12:56 +00:00
sprout::bind2nd(sprout::detail::char_traits_eq<char_traits>(), a)
)
),
s + n
);
2012-04-14 10:06:21 +00:00
}
static SPROUT_CXX14_CONSTEXPR char_type* move(char_type* s1, char_type const* s2, std::size_t n) {
2016-03-23 09:18:26 +00:00
sprout::copy_backward(s2, s2 + n, sprout::next(s1, n));
return s1;
2012-04-14 10:06:21 +00:00
}
static SPROUT_CXX14_CONSTEXPR char_type* copy(char_type* s1, char_type const* s2, std::size_t n) {
sprout::copy(s2, s2 + n, s1);
return s1;
2012-04-14 10:06:21 +00:00
}
static SPROUT_CXX14_CONSTEXPR char_type* assign(char_type* s, std::size_t n, char_type a) {
sprout::fill(s, s + n, a);
return s;
2012-04-14 10:06:21 +00:00
}
#ifdef SPROUT_NO_CXX11_CHAR_TRAITS
static SPROUT_CONSTEXPR int_type not_eof(int_type c) SPROUT_NOEXCEPT {
return eq_int_type(c, eof()) ? int_type()
: c
;
}
static SPROUT_CONSTEXPR char_type to_char_type(int_type c) SPROUT_NOEXCEPT {
return static_cast<char_type>(c);
}
static SPROUT_CONSTEXPR int_type to_int_type(char_type c) SPROUT_NOEXCEPT {
return static_cast<int_type>(c);
}
static SPROUT_CONSTEXPR bool eq_int_type(int_type c1, int_type c2) SPROUT_NOEXCEPT {
return c1 == c2;
}
static SPROUT_CONSTEXPR int_type eof() SPROUT_NOEXCEPT {
return static_cast<int_type>(EOF);
}
#else
2012-04-14 10:06:21 +00:00
static SPROUT_CONSTEXPR int_type not_eof(int_type c) SPROUT_NOEXCEPT {
return impl_type::not_eof(c);
}
static SPROUT_CONSTEXPR char_type to_char_type(int_type c) SPROUT_NOEXCEPT {
return impl_type::to_char_type(c);
}
static SPROUT_CONSTEXPR int_type to_int_type(char_type c) SPROUT_NOEXCEPT {
return impl_type::to_int_type(c);
}
static SPROUT_CONSTEXPR bool eq_int_type(int_type c1, int_type c2) SPROUT_NOEXCEPT {
return impl_type::eq_int_type(c1, c2);
}
static SPROUT_CONSTEXPR int_type eof() SPROUT_NOEXCEPT {
return impl_type::eof();
}
#endif
2012-04-14 10:06:21 +00:00
#if SPROUT_USE_INDEX_ITERATOR_IMPLEMENTATION
2016-03-23 09:18:26 +00:00
template<typename ForwardIterator>
static SPROUT_CONSTEXPR int compare(char_type const* s1, ForwardIterator s2, std::size_t n) {
2012-12-21 14:12:54 +00:00
return sprout::tristate_lexicographical_compare(
2013-02-26 08:03:30 +00:00
sprout::ptr_index(s1), sprout::ptr_index(s1, n), char_type(),
2016-03-23 09:18:26 +00:00
s2, sprout::next(s2, n), char_type(),
2013-02-19 16:12:56 +00:00
sprout::detail::char_traits_lt<char_traits>()
2012-12-21 14:12:54 +00:00
);
2012-04-14 10:06:21 +00:00
}
2016-03-23 09:18:26 +00:00
template<typename ForwardIterator>
static SPROUT_CONSTEXPR int compare(ForwardIterator s1, char_type const* s2, std::size_t n) {
2012-12-21 14:12:54 +00:00
return sprout::tristate_lexicographical_compare(
2016-03-23 09:18:26 +00:00
s1, sprout::next(s1, n), char_type(),
2013-02-26 08:03:30 +00:00
sprout::ptr_index(s2), sprout::ptr_index(s2, n), char_type(),
2013-02-19 16:12:56 +00:00
sprout::detail::char_traits_lt<char_traits>()
2012-12-21 14:12:54 +00:00
);
2012-04-14 10:06:21 +00:00
}
2016-03-23 09:18:26 +00:00
template<typename ForwardIterator1, typename ForwardIterator2>
static SPROUT_CONSTEXPR int compare(ForwardIterator1 s1, ForwardIterator2 s2, std::size_t n) {
2012-12-21 14:12:54 +00:00
return sprout::tristate_lexicographical_compare(
2016-03-23 09:18:26 +00:00
s1, sprout::next(s1, n), char_type(),
s2, sprout::next(s2, n), char_type(),
2013-02-19 16:12:56 +00:00
sprout::detail::char_traits_lt<char_traits>()
2012-12-21 14:12:54 +00:00
);
2012-04-14 10:06:21 +00:00
}
2016-03-23 09:18:26 +00:00
template<typename InputIterator>
static SPROUT_CONSTEXPR std::size_t length(InputIterator s) {
2013-01-12 16:16:48 +00:00
return sprout::detail::strlen(s);
2012-04-14 10:06:21 +00:00
}
2016-03-23 09:18:26 +00:00
template<typename ForwardIterator>
static SPROUT_CONSTEXPR ForwardIterator find(ForwardIterator s, std::size_t n, char_type const& a) {
2013-02-26 08:03:30 +00:00
return sprout::ptr_unindex(
2013-01-11 19:08:44 +00:00
sprout::find_if(
2016-03-23 09:18:26 +00:00
s, sprout::next(s, n),
2013-02-19 16:12:56 +00:00
sprout::bind2nd(sprout::detail::char_traits_eq<char_traits>(), a)
)
);
2012-04-14 10:06:21 +00:00
}
2016-03-23 09:18:26 +00:00
template<typename BidirectionalIterator1, typename BidirectionalIterator2>
static SPROUT_CXX14_CONSTEXPR BidirectionalIterator1 move(BidirectionalIterator1 s1, BidirectionalIterator2 s2, std::size_t n) {
sprout::copy_backward(s2, sprout::next(s2, n), sprout::next(s1, n));
2012-04-14 10:06:21 +00:00
return s1;
}
2016-03-23 09:18:26 +00:00
template<typename OutputIterator, typename BidirectionalIterator>
static SPROUT_CXX14_CONSTEXPR OutputIterator copy(OutputIterator s1, BidirectionalIterator s2, std::size_t n) {
sprout::copy(s2, sprout::next(s2, n), s1);
2012-04-14 10:06:21 +00:00
return s1;
}
2016-03-23 09:18:26 +00:00
template<typename ForwardIterator>
static SPROUT_CXX14_CONSTEXPR ForwardIterator assign(ForwardIterator s, std::size_t n, char_type a) {
sprout::fill(s, sprout::next(s, n), a);
2012-04-14 10:06:21 +00:00
return s;
}
#endif
};
//
// char_traits_helper
//
template<typename Traits>
class char_traits_helper {
public:
typedef Traits traits_type;
typedef typename traits_type::char_type char_type;
typedef typename traits_type::int_type int_type;
typedef typename traits_type::off_type off_type;
typedef typename traits_type::pos_type pos_type;
typedef typename traits_type::state_type state_type;
public:
static SPROUT_CONSTEXPR std::size_t length(char_type const* s, std::size_t n) {
2013-02-19 16:12:56 +00:00
return sprout::strlen(s, n);
}
static SPROUT_CONSTEXPR char_type const* find(char_type const* s, std::size_t n, char_type const& a) {
2013-02-26 08:03:30 +00:00
return sprout::ptr_unindex(
2013-01-11 19:08:44 +00:00
sprout::find_if(
2013-02-26 08:03:30 +00:00
sprout::ptr_index(s), sprout::ptr_index(s, n),
2013-02-19 16:12:56 +00:00
sprout::bind2nd(sprout::detail::char_traits_eq<traits_type>(), a)
)
);
}
static SPROUT_CONSTEXPR bool is_found(char_type const* found, char_type const* last) {
return found != last;
}
#if SPROUT_USE_INDEX_ITERATOR_IMPLEMENTATION
2016-03-23 09:18:26 +00:00
template<typename InputIterator>
static SPROUT_CONSTEXPR std::size_t length(InputIterator s, std::size_t n) {
2013-02-19 16:12:56 +00:00
return sprout::detail::strlen(s, n);
}
2016-03-23 09:18:26 +00:00
template<typename ForwardIterator>
static SPROUT_CONSTEXPR ForwardIterator find(ForwardIterator s, std::size_t n, char_type const& a) {
2013-01-11 19:08:44 +00:00
return sprout::find_if(
2016-03-23 09:18:26 +00:00
s, sprout::next(s, n),
2013-02-19 16:12:56 +00:00
sprout::bind2nd(sprout::detail::char_traits_eq<traits_type>(), a)
);
}
2016-03-23 09:18:26 +00:00
template<typename InputIterator>
static SPROUT_CONSTEXPR bool is_found(InputIterator found, InputIterator last) {
return found != last;
}
2012-04-14 10:06:21 +00:00
#endif
};
} // namespace sprout
#endif // #ifndef SPROUT_STRING_CHAR_TRAITS_HPP