#ifndef SPROUT_STRING_HPP #define SPROUT_STRING_HPP #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include HDR_ALGORITHM_SSCRISK_CEL_OR_SPROUT_DETAIL namespace sprout { // // char_traits // template class char_traits { private: typedef std::char_traits 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; public: static void assign(char_type& c1, char_type const& c2) SPROUT_NOEXCEPT { impl_type::assign(c1, c2); } 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); } static SPROUT_CONSTEXPR int compare(char_type const* s1, char_type const* s2, std::size_t n) { return !n ? 0 : lt(*s1, *s2) ? -1 : lt(*s2, *s1) ? 1 : compare(s1 + 1, s2 + 1, n - 1) ; } static SPROUT_CONSTEXPR std::size_t length(char_type const* s) { return !*s ? 0 : 1 + length(s + 1) ; } static SPROUT_CONSTEXPR char_type const* find(char_type const* s, std::size_t n, char_type const& a) { return !n ? nullptr : eq(*s, a) ? s : find(s + 1, n - 1, a) ; } static char_type* move(char_type* s1, char_type const* s2, std::size_t n) { return impl_type::move(s1, s2, n); } static char_type* copy(char_type* s1, char_type const* s2, std::size_t n) { return impl_type::copy(s1, s2, n); } static char_type* assign(char_type* s, std::size_t n, char_type a) { return impl_type::assign(s, n, a); } 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(); } }; // // basic_string // template > class basic_string { 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; typedef sprout::reverse_iterator reverse_iterator; typedef sprout::reverse_iterator const_reverse_iterator; typedef Traits traits_type; public: SPROUT_STATIC_CONSTEXPR size_type npos = -1; SPROUT_STATIC_CONSTEXPR size_type static_size = N; SPROUT_STATIC_CONSTEXPR size_type fixed_size = static_size; private: static SPROUT_CONSTEXPR int compare_impl_2(int compared, size_type n1, size_type n2) { return compared != 0 ? compared : n1 < n2 ? -1 : n2 < n1 ? 1 : 0 ; } static SPROUT_CONSTEXPR int compare_impl_1(value_type const* dest, size_type pos1, size_type n1, value_type const* s, size_type n2) { return compare_impl_2( traits_type::compare(dest + pos1, s, NS_SSCRISK_CEL_OR_SPROUT_DETAIL::min(n1, n2)), n1, n2 ); } template static SPROUT_CONSTEXPR basic_string from_c_str_impl( value_type const* s, size_type n, sprout::index_tuple ) { return sprout::basic_string{{(Indexes < n ? s[Indexes] : T())...}, n}; } public: static SPROUT_CONSTEXPR basic_string from_c_str(value_type const* s, size_type n) { return !(N < n) ? from_c_str_impl(s, n, typename sprout::index_range<0, N>::type()) : throw "basic_string<>: index out of range" ; } static SPROUT_CONSTEXPR basic_string from_c_str(value_type const* s) { return from_c_str(s, traits_type::length(s)); } public: T elems[N + 1]; size_type len; public: SPROUT_CONSTEXPR size_type size() const SPROUT_NOEXCEPT { return len; } SPROUT_CONSTEXPR size_type length() const SPROUT_NOEXCEPT { return size(); } SPROUT_CONSTEXPR bool empty() const SPROUT_NOEXCEPT { return size() == 0; } SPROUT_CONSTEXPR size_type max_size() const SPROUT_NOEXCEPT { return N; } void rangecheck(size_type i) const { if (i >= size()) { throw std::out_of_range("basic_string<>: index out of range"); } } void maxcheck(size_type n) const { if (n > max_size()) { throw std::out_of_range("basic_string<>: index out of range"); } } iterator begin() SPROUT_NOEXCEPT { return &elems[0]; } SPROUT_CONSTEXPR const_iterator begin() const SPROUT_NOEXCEPT { return &elems[0]; } SPROUT_CONSTEXPR const_iterator cbegin() const SPROUT_NOEXCEPT { return &elems[0]; } iterator end() SPROUT_NOEXCEPT { return &elems[0] + size(); } SPROUT_CONSTEXPR const_iterator end() const SPROUT_NOEXCEPT { return &elems[0] + size(); } SPROUT_CONSTEXPR const_iterator cend() const SPROUT_NOEXCEPT { return &elems[0] + size(); } reverse_iterator rbegin() SPROUT_NOEXCEPT { return const_reverse_iterator(end()); } SPROUT_CONSTEXPR const_reverse_iterator rbegin() const SPROUT_NOEXCEPT { return const_reverse_iterator(end()); } SPROUT_CONSTEXPR const_reverse_iterator crbegin() const SPROUT_NOEXCEPT { return const_reverse_iterator(end()); } reverse_iterator rend() SPROUT_NOEXCEPT { return const_reverse_iterator(begin()); } SPROUT_CONSTEXPR const_reverse_iterator rend() const SPROUT_NOEXCEPT { return const_reverse_iterator(begin()); } SPROUT_CONSTEXPR const_reverse_iterator crend() const SPROUT_NOEXCEPT { return const_reverse_iterator(begin()); } reference operator[](size_type i) { return elems[i]; } SPROUT_CONSTEXPR const_reference operator[](size_type i) const { return elems[i]; } reference at(size_type i) { rangecheck(i); return elems[i]; } const_reference at(size_type i) const { rangecheck(i); return elems[i]; } reference front() { return elems[0]; } SPROUT_CONSTEXPR const_reference front() const { return elems[0]; } reference back() { return elems[size() - 1]; } SPROUT_CONSTEXPR const_reference back() const { return elems[size() - 1]; } pointer data() SPROUT_NOEXCEPT { return &elems[0]; } SPROUT_CONSTEXPR const_pointer data() const SPROUT_NOEXCEPT { return &elems[0]; } pointer c_array() SPROUT_NOEXCEPT { return &elems[0]; } SPROUT_CONSTEXPR const_pointer c_str() const SPROUT_NOEXCEPT { return &elems[0]; } void resize(size_type n, value_type c) { maxcheck(n); if (n > size()) { traits_type::assign(end(), n - size(), c); } traits_type::assign(begin() + n, max_size() - n, value_type()); len = n; } void resize(size_type n) { resize(n, value_type()); } void clear() { traits_type::assign(begin(), max_size(), value_type()); len = 0; } template basic_string& assign(basic_string const& str) { return assign(str.begin(), str.size()); } template basic_string& assign(basic_string const& str, size_type pos, size_type n) { if (str.size() < pos + n) { throw std::out_of_range("basic_string<>: index out of range"); } return assign(str.begin() + pos, n); } basic_string& assign(value_type const* s, size_type n) { maxcheck(n); for (size_type i = 0; i < n; ++i) { traits_type::assign(elems[i], s[i]); } for (size_type i = n; i < max_size(); ++i) { traits_type::assign(elems[i], value_type()); } len = n; return *this; } basic_string& assign(value_type const* s) { return assign(s, traits_type::length(s)); } basic_string& assign(size_type n, value_type c) { maxcheck(n); traits_type::assign(begin(), n, c); traits_type::assign(begin() + n, max_size() - n, value_type()); len = n; return *this; } template basic_string& assign(Iterator first, Iterator last) { size_type n = 0; for (; n < max_size() || first != last; ++n, ++first) { traits_type::assign(elems[n], *first); } for (size_type i = n; i < max_size(); ++i) { traits_type::assign(elems[i], value_type()); } len = n; return *this; } void swap(basic_string& other) SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(std::swap(std::declval(), std::declval()))) { std::swap_ranges(other.begin(), other.begin() + other.max_size(), begin()); { using std::swap; swap(len, other.len); } } template basic_string& operator=(basic_string const& rhs) { return assign(rhs); } basic_string& operator=(value_type const* rhs) { return assign(rhs); } basic_string& operator=(value_type rhs) { return assign(1, rhs); } template SPROUT_CONSTEXPR int compare(basic_string const& str) const { return compare(0, size(), str.c_str(), str.size()); } SPROUT_CONSTEXPR int compare(T const* s) const { return compare(0, size(), s, traits_type::length(s)); } template SPROUT_CONSTEXPR int compare(size_type pos1, size_type n1, basic_string const& str) const { return compare(pos1, n1, str, 0, npos); } SPROUT_CONSTEXPR int compare(size_type pos1, size_type n1, char const* s) const { return compare(pos1, n1, s, traits_type::length(s)); } template SPROUT_CONSTEXPR int compare(size_type pos1, size_type n1, basic_string const& str, size_type pos2, size_type n2) const { return !(str.size() < pos2) ? compare(pos1, n1, str.c_str() + pos2, NS_SSCRISK_CEL_OR_SPROUT_DETAIL::min(n2, str.size() - pos2)) : throw "basic_string<>: index out of range" ; } SPROUT_CONSTEXPR int compare(size_type pos1, size_type n1, char const* s, size_type n2) const { return !(size() < pos1) ? compare_impl_1(c_str(), pos1, NS_SSCRISK_CEL_OR_SPROUT_DETAIL::min(n1, size() - pos1), s, n2) : throw "basic_string<>: index out of range" ; } SPROUT_CONSTEXPR basic_string substr(size_type pos = 0, size_type n = npos) const { return !(size() < pos) ? n == npos ? substr(pos, size() - pos) : from_c_str(c_str() + pos, n) : throw "basic_string<>: index out of range" ; } }; template SPROUT_CONSTEXPR inline bool operator==(sprout::basic_string const& lhs, sprout::basic_string const& rhs) { return lhs.compare(rhs) == 0; } template SPROUT_CONSTEXPR inline bool operator==(sprout::basic_string const& lhs, T const* rhs) { return lhs.compare(rhs) == 0; } template SPROUT_CONSTEXPR inline bool operator==(T const* lhs, sprout::basic_string const& rhs) { return 0 == rhs.compare(lhs); } template SPROUT_CONSTEXPR inline bool operator!=(sprout::basic_string const& lhs, sprout::basic_string const& rhs) { return !(lhs == rhs); } template SPROUT_CONSTEXPR inline bool operator!=(sprout::basic_string const& lhs, T const* rhs) { return !(lhs == rhs); } template SPROUT_CONSTEXPR inline bool operator!=(T const* lhs, sprout::basic_string const& rhs) { return !(lhs == rhs); } template SPROUT_CONSTEXPR inline bool operator<(sprout::basic_string const& lhs, sprout::basic_string const& rhs) { return lhs.compare(rhs) < 0; } template SPROUT_CONSTEXPR inline bool operator<(sprout::basic_string const& lhs, T const* rhs) { return lhs.compare(rhs) < 0; } template SPROUT_CONSTEXPR inline bool operator<(T const* lhs, sprout::basic_string const& rhs) { return 0 < rhs.compare(lhs); } template SPROUT_CONSTEXPR inline bool operator>(sprout::basic_string const& lhs, sprout::basic_string const& rhs) { return rhs < lhs; } template SPROUT_CONSTEXPR inline bool operator>(sprout::basic_string const& lhs, T const* rhs) { return rhs < lhs; } template SPROUT_CONSTEXPR inline bool operator>(T const* lhs, sprout::basic_string const& rhs) { return rhs < lhs; } template SPROUT_CONSTEXPR inline bool operator<=(sprout::basic_string const& lhs, sprout::basic_string const& rhs) { return !(rhs < lhs); } template SPROUT_CONSTEXPR inline bool operator<=(sprout::basic_string const& lhs, T const* rhs) { return !(rhs < lhs); } template SPROUT_CONSTEXPR inline bool operator<=(T const* lhs, sprout::basic_string const& rhs) { return !(rhs < lhs); } template SPROUT_CONSTEXPR inline bool operator>=(sprout::basic_string const& lhs, sprout::basic_string const& rhs) { return !(lhs < rhs); } template SPROUT_CONSTEXPR inline bool operator>=(sprout::basic_string const& lhs, T const* rhs) { return !(lhs < rhs); } template SPROUT_CONSTEXPR inline bool operator>=(T const* lhs, sprout::basic_string const& rhs) { return !(lhs < rhs); } // // swap // template inline void swap( sprout::basic_string& lhs, sprout::basic_string& rhs ) SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(lhs.swap(rhs))) { lhs.swap(rhs); } // // rebind_fixed_size // template struct rebind_fixed_size > { public: template >::size_type S> struct apply { public: typedef sprout::basic_string type; }; }; namespace detail { template struct make_clone_functor_impl; template struct make_clone_functor_impl > { private: typedef sprout::basic_string container_type; typedef typename sprout::fixed_container_traits::clone_type clone_type; private: template static SPROUT_CONSTEXPR clone_type make_impl(typename clone_type::size_type size) { return clone_type{{}, size}; } template static SPROUT_CONSTEXPR typename std::enable_if< S != sizeof...(Args), clone_type >::type make_impl(typename clone_type::size_type size, T const& head, Args const&... tail) { return make_impl(size, tail..., S < size ? head : T()); } template static SPROUT_CONSTEXPR typename std::enable_if< S == sizeof...(Args), clone_type >::type make_impl(typename clone_type::size_type size, T const& head, Args const&... tail) { return clone_type{{tail..., head}, size}; } public: static SPROUT_CONSTEXPR typename clone_type::size_type length() { return 0; } template static SPROUT_CONSTEXPR typename clone_type::size_type length(T const& head, Tail const&... tail) { return !head ? 0 : 1 + length(tail...); } template static SPROUT_CONSTEXPR clone_type make(typename clone_type::size_type size, Args const&... args) { return make_impl<0>(size, args...); } }; } // namespace detail // // make_clone_functor // template struct make_clone_functor > { private: typedef sprout::basic_string container_type; typedef sprout::detail::make_clone_functor_impl impl_type; public: template SPROUT_CONSTEXPR typename sprout::fixed_container_traits::clone_type operator()(Args const&... args) const { return impl_type::make(impl_type::length(args...), args...); } }; // // remake_clone_functor // template struct remake_clone_functor > { private: typedef sprout::basic_string container_type; typedef sprout::detail::make_clone_functor_impl impl_type; public: template SPROUT_CONSTEXPR typename sprout::fixed_container_traits::clone_type operator()( Other& other, typename sprout::fixed_container_traits::difference_type size, Args const&... args ) const { return impl_type::make(size, args...); } template SPROUT_CONSTEXPR typename sprout::fixed_container_traits::clone_type operator()( Other const& other, typename sprout::fixed_container_traits::difference_type size, Args const&... args ) const { return impl_type::make(size, args...); } }; namespace detail { template struct is_basic_string_impl { public: typedef std::integral_constant type; SPROUT_STATIC_CONSTEXPR bool value = type::value; }; template struct is_basic_string_impl< T, typename std::enable_if< std::is_same< T, sprout::basic_string >::value >::type > { public: typedef std::integral_constant type; SPROUT_STATIC_CONSTEXPR bool value = type::value; }; } // namespace detail // // is_basic_string // template struct is_basic_string : public sprout::detail::is_basic_string_impl {}; namespace detail { template struct is_string_of_impl { public: typedef std::integral_constant type; SPROUT_STATIC_CONSTEXPR bool value = type::value; }; template struct is_string_of_impl< T, Elem, typename std::enable_if< std::is_same< T, sprout::basic_string >::value >::type > { public: typedef std::integral_constant type; SPROUT_STATIC_CONSTEXPR bool value = type::value; }; } // namespace detail // // is_string_of // template struct is_string_of : public sprout::detail::is_string_of_impl {}; // // is_string // template struct is_string : public sprout::is_string_of {}; // // is_wstring // template struct is_wstring : public sprout::is_string_of {}; // // is_u16string // template struct is_u16string : public sprout::is_string_of {}; // // is_u32string // template struct is_u32string : public sprout::is_string_of {}; namespace detail { template SPROUT_CONSTEXPR inline sprout::basic_string to_string_impl_1( T const(& arr)[N], typename sprout::basic_string::size_type n, sprout::index_tuple ) { return sprout::basic_string{{(Indexes < n ? arr[Indexes] : T())...}, n}; } template SPROUT_CONSTEXPR inline sprout::basic_string to_string_impl( T const(& arr)[N], sprout::index_tuple ) { return to_string_impl_1(arr, sprout::char_traits::length(arr), sprout::index_tuple()); } } // namespace detail // // to_string // template SPROUT_CONSTEXPR inline sprout::basic_string to_string(T const(& arr)[N]) { return sprout::detail::to_string_impl(arr, typename sprout::index_range<0, N - 1>::type()); } // // string_from_c_str // template SPROUT_CONSTEXPR inline sprout::basic_string string_from_c_str(T const* s, std::size_t n) { return sprout::basic_string::from_c_str(s, n); } template SPROUT_CONSTEXPR inline sprout::basic_string string_from_c_str(T const* s) { return sprout::basic_string::from_c_str(s); } template SPROUT_CONSTEXPR inline typename sprout::fixed::result_of::push_back, T>::type operator+( sprout::basic_string const& lhs, T const& rhs ) { return sprout::fixed::push_back(lhs, rhs); } template SPROUT_CONSTEXPR inline typename sprout::fixed::result_of::push_front, T>::type operator+( T const& lhs, sprout::basic_string const& rhs ) { return sprout::fixed::push_front(rhs, lhs); } template SPROUT_CONSTEXPR inline typename sprout::fixed::result_of::append_back< sprout::basic_string, sprout::basic_string >::type operator+( sprout::basic_string const& lhs, T const (& rhs)[N2] ) { return sprout::fixed::append_back(lhs, sprout::to_string(rhs)); } template SPROUT_CONSTEXPR inline typename sprout::fixed::result_of::append_front< sprout::basic_string, sprout::basic_string >::type operator+( T const (& lhs)[N2], sprout::basic_string const& rhs ) { return sprout::fixed::append_front(rhs, sprout::to_string(lhs)); } template SPROUT_CONSTEXPR inline typename sprout::fixed::result_of::append_back< sprout::basic_string, sprout::basic_string >::type operator+( sprout::basic_string const& lhs, sprout::basic_string const& rhs ) { return sprout::fixed::append_back(lhs, rhs); } template inline std::basic_ostream& operator<<(std::basic_ostream& lhs, sprout::basic_string const& rhs) { return lhs << rhs.c_str(); } template inline std::basic_istream& operator>>(std::basic_istream& lhs, sprout::basic_string& rhs) { typedef T elem_type; typedef StreamTraits traits_type; typedef std::basic_istream istream_type; typedef sprout::basic_string string_type; typedef std::ctype ctype_type; typedef typename string_type::size_type size_type; std::ios_base::iostate state = std::ios_base::goodbit; bool changed = false; size_type current = 0; if (typename istream_type::sentry(lhs)) { ctype_type const& ctype_fac = std::use_facet(lhs.getloc()); try { size_type remain = 0 < lhs.width() && static_cast(lhs.width()) < rhs.max_size() ? static_cast(lhs.width()) : rhs.max_size() ; typename traits_type::int_type meta = lhs.rdbuf()->sgetc(); for (; remain; --remain, meta = lhs.rdbuf()->snextc()) if (traits_type::eq_int_type(traits_type::eof(), meta)) { state |= std::ios_base::eofbit; break; } else if (ctype_fac.is(ctype_type::space, traits_type::to_char_type(meta))) { break; } else { rhs[current] = traits_type::to_char_type(meta); changed = true; ++current; } } catch (...) { state |= std::ios_base::badbit; } } lhs.width(0); if (!changed) { state |= std::ios_base::failbit; } lhs.setstate(state); rhs.len = current; for (; current != rhs.max_size(); ++current) { rhs[current] = T(); } return lhs; } #if 0 // // string // template using string = sprout::basic_string; // // wstring // template using wstring = sprout::basic_string; // // u16string // template using u16string = sprout::basic_string; // // u32string // template using u32string = sprout::basic_string; #endif // #if 0 // // string_t // template struct string_t { public: typedef sprout::basic_string type; }; // // wstring_t // template struct wstring_t { public: typedef sprout::basic_string type; }; // // u16string_t // template struct u16string_t { public: typedef sprout::basic_string type; }; // // u32string_t // template struct u32string_t { public: typedef sprout::basic_string type; }; } // namespace sprout namespace std { // // tuple_size // template struct tuple_size > { public: typedef std::integral_constant type; SPROUT_STATIC_CONSTEXPR std::size_t value = type::value; }; // // tuple_element // template struct tuple_element > { public: static_assert(I < N, "tuple_element<>: index out of range"); typedef T type; }; // // get // template T& get(sprout::basic_string& arr) SPROUT_NOEXCEPT { static_assert(I < N, "get: index out of range"); return arr[I]; } template SPROUT_CONSTEXPR T const& get(sprout::basic_string const& arr) SPROUT_NOEXCEPT { static_assert(I < N, "get: index out of range"); return arr[I]; } template T&& get(sprout::basic_string&& arr) SPROUT_NOEXCEPT { return std::move(std::get(arr)); } } // namespace std #endif // #ifndef SPROUT_STRING_HPP