string.hpp 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290
  1. /*!
  2. @file
  3. Defines `boost::hana::string`.
  4. @copyright Louis Dionne 2013-2016
  5. Distributed under the Boost Software License, Version 1.0.
  6. (See accompanying file LICENSE.md or copy at http://boost.org/LICENSE_1_0.txt)
  7. */
  8. #ifndef BOOST_HANA_STRING_HPP
  9. #define BOOST_HANA_STRING_HPP
  10. #include <boost/hana/fwd/string.hpp>
  11. #include <boost/hana/bool.hpp>
  12. #include <boost/hana/config.hpp>
  13. #include <boost/hana/core/make.hpp>
  14. #include <boost/hana/detail/algorithm.hpp>
  15. #include <boost/hana/detail/operators/adl.hpp>
  16. #include <boost/hana/detail/operators/comparable.hpp>
  17. #include <boost/hana/detail/operators/iterable.hpp>
  18. #include <boost/hana/detail/operators/orderable.hpp>
  19. #include <boost/hana/fwd/at.hpp>
  20. #include <boost/hana/fwd/contains.hpp>
  21. #include <boost/hana/fwd/core/tag_of.hpp>
  22. #include <boost/hana/fwd/core/to.hpp>
  23. #include <boost/hana/fwd/drop_front.hpp>
  24. #include <boost/hana/fwd/equal.hpp>
  25. #include <boost/hana/fwd/find.hpp>
  26. #include <boost/hana/fwd/front.hpp>
  27. #include <boost/hana/fwd/hash.hpp>
  28. #include <boost/hana/fwd/is_empty.hpp>
  29. #include <boost/hana/fwd/length.hpp>
  30. #include <boost/hana/fwd/less.hpp>
  31. #include <boost/hana/fwd/unpack.hpp>
  32. #include <boost/hana/if.hpp>
  33. #include <boost/hana/integral_constant.hpp>
  34. #include <boost/hana/optional.hpp>
  35. #include <boost/hana/type.hpp>
  36. #include <utility>
  37. #include <cstddef>
  38. #include <type_traits>
  39. BOOST_HANA_NAMESPACE_BEGIN
  40. //////////////////////////////////////////////////////////////////////////
  41. // string<>
  42. //////////////////////////////////////////////////////////////////////////
  43. //! @cond
  44. template <char ...s>
  45. struct string
  46. : detail::operators::adl<string<s...>>
  47. , detail::iterable_operators<string<s...>>
  48. { };
  49. //! @endcond
  50. template <char ...s>
  51. struct tag_of<string<s...>> {
  52. using type = string_tag;
  53. };
  54. //////////////////////////////////////////////////////////////////////////
  55. // make<string_tag>
  56. //////////////////////////////////////////////////////////////////////////
  57. template <>
  58. struct make_impl<string_tag> {
  59. template <typename ...Chars>
  60. static constexpr auto apply(Chars const& ...) {
  61. return hana::string<hana::value<Chars>()...>{};
  62. }
  63. };
  64. //////////////////////////////////////////////////////////////////////////
  65. // BOOST_HANA_STRING
  66. //////////////////////////////////////////////////////////////////////////
  67. namespace string_detail {
  68. template <typename S, std::size_t ...N>
  69. constexpr string<S::get()[N]...>
  70. prepare_impl(S, std::index_sequence<N...>)
  71. { return {}; }
  72. template <typename S>
  73. constexpr decltype(auto) prepare(S s) {
  74. return prepare_impl(s,
  75. std::make_index_sequence<sizeof(S::get()) - 1>{});
  76. }
  77. }
  78. #define BOOST_HANA_STRING(s) \
  79. (::boost::hana::string_detail::prepare([]{ \
  80. struct tmp { \
  81. static constexpr decltype(auto) get() { return s; } \
  82. }; \
  83. return tmp{}; \
  84. }())) \
  85. /**/
  86. #ifdef BOOST_HANA_CONFIG_ENABLE_STRING_UDL
  87. //////////////////////////////////////////////////////////////////////////
  88. // _s user-defined literal
  89. //////////////////////////////////////////////////////////////////////////
  90. namespace literals {
  91. template <typename CharT, CharT ...s>
  92. constexpr auto operator"" _s() {
  93. static_assert(std::is_same<CharT, char>::value,
  94. "hana::string: Only narrow string literals are supported with "
  95. "the _s string literal right now. See https://goo.gl/fBbKD7 "
  96. "if you need support for fancier types of compile-time strings.");
  97. return hana::string_c<s...>;
  98. }
  99. }
  100. #endif
  101. //////////////////////////////////////////////////////////////////////////
  102. // Operators
  103. //////////////////////////////////////////////////////////////////////////
  104. namespace detail {
  105. template <>
  106. struct comparable_operators<string_tag> {
  107. static constexpr bool value = true;
  108. };
  109. template <>
  110. struct orderable_operators<string_tag> {
  111. static constexpr bool value = true;
  112. };
  113. }
  114. //////////////////////////////////////////////////////////////////////////
  115. // to<char const*>
  116. //////////////////////////////////////////////////////////////////////////
  117. template <>
  118. struct to_impl<char const*, string_tag> {
  119. template <char ...c>
  120. static constexpr char const c_string[sizeof...(c) + 1] = {c..., '\0'};
  121. template <char ...c>
  122. static constexpr char const* apply(string<c...> const&)
  123. { return c_string<c...>; }
  124. };
  125. template <char ...c>
  126. constexpr char const to_impl<char const*, string_tag>::c_string[sizeof...(c) + 1];
  127. //////////////////////////////////////////////////////////////////////////
  128. // Comparable
  129. //////////////////////////////////////////////////////////////////////////
  130. template <>
  131. struct equal_impl<string_tag, string_tag> {
  132. template <typename S>
  133. static constexpr auto apply(S const&, S const&)
  134. { return hana::true_c; }
  135. template <typename S1, typename S2>
  136. static constexpr auto apply(S1 const&, S2 const&)
  137. { return hana::false_c; }
  138. };
  139. //////////////////////////////////////////////////////////////////////////
  140. // Orderable
  141. //////////////////////////////////////////////////////////////////////////
  142. template <>
  143. struct less_impl<string_tag, string_tag> {
  144. template <char ...s1, char ...s2>
  145. static constexpr auto
  146. apply(string<s1...> const&, string<s2...> const&) {
  147. // We put a '\0' at the end only to avoid empty arrays.
  148. constexpr char const c_str1[] = {s1..., '\0'};
  149. constexpr char const c_str2[] = {s2..., '\0'};
  150. return hana::bool_c<detail::lexicographical_compare(
  151. c_str1, c_str1 + sizeof...(s1),
  152. c_str2, c_str2 + sizeof...(s2)
  153. )>;
  154. }
  155. };
  156. //////////////////////////////////////////////////////////////////////////
  157. // Foldable
  158. //////////////////////////////////////////////////////////////////////////
  159. template <>
  160. struct unpack_impl<string_tag> {
  161. template <char ...s, typename F>
  162. static constexpr decltype(auto) apply(string<s...> const&, F&& f)
  163. { return static_cast<F&&>(f)(char_<s>{}...); }
  164. };
  165. template <>
  166. struct length_impl<string_tag> {
  167. template <char ...s>
  168. static constexpr auto apply(string<s...> const&)
  169. { return hana::size_c<sizeof...(s)>; }
  170. };
  171. //////////////////////////////////////////////////////////////////////////
  172. // Iterable
  173. //////////////////////////////////////////////////////////////////////////
  174. template <>
  175. struct front_impl<string_tag> {
  176. template <char x, char ...xs>
  177. static constexpr auto apply(string<x, xs...> const&)
  178. { return hana::char_c<x>; }
  179. };
  180. template <>
  181. struct drop_front_impl<string_tag> {
  182. template <std::size_t N, char ...xs, std::size_t ...i>
  183. static constexpr auto helper(string<xs...> const&, std::index_sequence<i...>) {
  184. constexpr char s[] = {xs...};
  185. return hana::string_c<s[i + N]...>;
  186. }
  187. template <char ...xs, typename N>
  188. static constexpr auto apply(string<xs...> const& s, N const&) {
  189. return helper<N::value>(s, std::make_index_sequence<
  190. N::value < sizeof...(xs) ? sizeof...(xs) - N::value : 0
  191. >{});
  192. }
  193. template <typename N>
  194. static constexpr auto apply(string<> const& s, N const&)
  195. { return s; }
  196. };
  197. template <>
  198. struct is_empty_impl<string_tag> {
  199. template <char ...s>
  200. static constexpr auto apply(string<s...> const&)
  201. { return hana::bool_c<sizeof...(s) == 0>; }
  202. };
  203. template <>
  204. struct at_impl<string_tag> {
  205. template <char ...s, typename N>
  206. static constexpr auto apply(string<s...> const&, N const&) {
  207. // We put a '\0' at the end to avoid an empty array.
  208. constexpr char characters[] = {s..., '\0'};
  209. constexpr auto n = N::value;
  210. return hana::char_c<characters[n]>;
  211. }
  212. };
  213. //////////////////////////////////////////////////////////////////////////
  214. // Searchable
  215. //////////////////////////////////////////////////////////////////////////
  216. template <>
  217. struct contains_impl<string_tag> {
  218. template <char ...s, typename C>
  219. static constexpr auto
  220. helper(string<s...> const&, C const&, hana::true_) {
  221. constexpr char const characters[] = {s..., '\0'};
  222. constexpr char c = hana::value<C>();
  223. return hana::bool_c<
  224. detail::find(characters, characters + sizeof...(s), c)
  225. != characters + sizeof...(s)
  226. >;
  227. }
  228. template <typename S, typename C>
  229. static constexpr auto helper(S const&, C const&, hana::false_)
  230. { return hana::false_c; }
  231. template <typename S, typename C>
  232. static constexpr auto apply(S const& s, C const& c)
  233. { return helper(s, c, hana::bool_c<hana::Constant<C>::value>); }
  234. };
  235. template <>
  236. struct find_impl<string_tag> {
  237. template <char ...s, typename Char>
  238. static constexpr auto apply(string<s...> const& str, Char const& c) {
  239. return hana::if_(contains_impl<string_tag>::apply(str, c),
  240. hana::just(c),
  241. hana::nothing
  242. );
  243. }
  244. };
  245. //////////////////////////////////////////////////////////////////////////
  246. // Hashable
  247. //////////////////////////////////////////////////////////////////////////
  248. template <>
  249. struct hash_impl<string_tag> {
  250. template <typename String>
  251. static constexpr auto apply(String const&) {
  252. return hana::type_c<String>;
  253. }
  254. };
  255. BOOST_HANA_NAMESPACE_END
  256. #endif // !BOOST_HANA_STRING_HPP