StringUtilities.cpp 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253
  1. /*
  2. * This source file is part of RmlUi, the HTML/CSS Interface Middleware
  3. *
  4. * For the latest information, see http://github.com/mikke89/RmlUi
  5. *
  6. * Copyright (c) 2008-2010 CodePoint Ltd, Shift Technology Ltd
  7. * Copyright (c) 2019-2023 The RmlUi Team, and contributors
  8. *
  9. * Permission is hereby granted, free of charge, to any person obtaining a copy
  10. * of this software and associated documentation files (the "Software"), to deal
  11. * in the Software without restriction, including without limitation the rights
  12. * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  13. * copies of the Software, and to permit persons to whom the Software is
  14. * furnished to do so, subject to the following conditions:
  15. *
  16. * The above copyright notice and this permission notice shall be included in
  17. * all copies or substantial portions of the Software.
  18. *
  19. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  20. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  21. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  22. * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  23. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  24. * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  25. * THE SOFTWARE.
  26. *
  27. */
  28. #include "../Common/TypesToString.h"
  29. #include <RmlUi/Core/StringUtilities.h>
  30. #include <RmlUi/Core/Types.h>
  31. #include <doctest.h>
  32. using namespace Rml;
  33. TEST_CASE("StringUtilities::TrimTrailingDotZeros")
  34. {
  35. auto RunTrimTrailingDotZeros = [](String string) {
  36. StringUtilities::TrimTrailingDotZeros(string);
  37. return string;
  38. };
  39. CHECK(RunTrimTrailingDotZeros("0.1") == "0.1");
  40. CHECK(RunTrimTrailingDotZeros("0.10") == "0.1");
  41. CHECK(RunTrimTrailingDotZeros("0.1000") == "0.1");
  42. CHECK(RunTrimTrailingDotZeros("0.01") == "0.01");
  43. CHECK(RunTrimTrailingDotZeros("0.") == "0");
  44. CHECK(RunTrimTrailingDotZeros("5.") == "5");
  45. CHECK(RunTrimTrailingDotZeros("5.5") == "5.5");
  46. CHECK(RunTrimTrailingDotZeros("5.50") == "5.5");
  47. CHECK(RunTrimTrailingDotZeros("5.501") == "5.501");
  48. CHECK(RunTrimTrailingDotZeros("10.0") == "10");
  49. CHECK(RunTrimTrailingDotZeros("11.0") == "11");
  50. // Some test cases for behavior that are probably not what you want.
  51. WARN(RunTrimTrailingDotZeros("test0") == "test");
  52. WARN(RunTrimTrailingDotZeros("1000") == "1");
  53. WARN(RunTrimTrailingDotZeros(".") == "");
  54. WARN(RunTrimTrailingDotZeros("0") == "");
  55. WARN(RunTrimTrailingDotZeros(".0") == "");
  56. WARN(RunTrimTrailingDotZeros(" 11 2121 3.00") == " 11 2121 3");
  57. WARN(RunTrimTrailingDotZeros("11") == "11");
  58. }
  59. TEST_CASE("StringUtilities::ExpandString")
  60. {
  61. auto ExpandStringShort = [](const String string, char delimiter, bool ignore_repeated_delimiters = false) {
  62. StringList result;
  63. StringUtilities::ExpandString(result, string, delimiter, ignore_repeated_delimiters);
  64. return StringListWrapper{result};
  65. };
  66. auto ExpandStringLong = [](const String string, char delimiter, char quote, char unquote, bool ignore_repeated_delimiters) {
  67. StringList result;
  68. StringUtilities::ExpandString(result, string, delimiter, quote, unquote, ignore_repeated_delimiters);
  69. return StringListWrapper{result};
  70. };
  71. auto ExpandString = [&](const String string, char delimiter, bool ignore_repeated_delimiters = false) {
  72. const StringListWrapper short_result = ExpandStringShort(string, delimiter, ignore_repeated_delimiters);
  73. const StringListWrapper long_result = ExpandStringLong(string, delimiter, '(', ')', ignore_repeated_delimiters);
  74. CAPTURE(string);
  75. CAPTURE(String{delimiter});
  76. CHECK(short_result == long_result);
  77. return short_result;
  78. };
  79. CHECK(ExpandString("a,b,c", ',') == StringListWrapper{{"a", "b", "c"}});
  80. CHECK(ExpandString("a,b,c", ' ') == StringListWrapper{{"a,b,c"}});
  81. CHECK(ExpandString("a b c", ' ') == StringListWrapper{{"a", "b", "c"}});
  82. CHECK(ExpandString("a,b,,c", ',') == StringListWrapper{{"a", "b", "", "c"}});
  83. CHECK(ExpandString(" a , b , c ", ',') == StringListWrapper{{"a", "b", "c"}});
  84. CHECK(ExpandString("a", ',') == StringListWrapper{{"a"}});
  85. CHECK(ExpandStringShort("", ',') == StringListWrapper{{""}});
  86. CHECK(ExpandStringShort("", ' ') == StringListWrapper{{""}});
  87. CHECK(ExpandStringShort(" ", ' ') == StringListWrapper{{""}});
  88. CHECK(ExpandString(",a,b", ',') == StringListWrapper{{"", "a", "b"}});
  89. CHECK(ExpandString("a,,b", ',') == StringListWrapper{{"a", "", "b"}});
  90. CHECK(ExpandString("a;b;c", ';') == StringListWrapper{{"a", "b", "c"}});
  91. CHECK(ExpandStringLong("(a,b),c", ',', '(', ')', false) == StringListWrapper{{"(a,b)", "c"}});
  92. CHECK(ExpandStringLong("((a,b),c),d", ',', '(', ')', false) == StringListWrapper{{"((a,b),c)", "d"}});
  93. CHECK(ExpandStringLong("a,,b", ',', '(', ')', false) == StringListWrapper{{"a", "", "b"}});
  94. CHECK(ExpandStringLong("a,,b", ',', '(', ')', true) == StringListWrapper{{"a", "b"}});
  95. CHECK(ExpandStringLong("a b", ' ', '(', ')', false) == StringListWrapper{{"a", "", "b"}});
  96. CHECK(ExpandStringLong("a b", ' ', '(', ')', true) == StringListWrapper{{"a", "b"}});
  97. // Questionable behavior, consider changing implementation to match the suggestion.
  98. CHECK(ExpandString("a,b,", ',') == StringListWrapper{{"a", "b"}}); // { "a", "b", ""}
  99. CHECK(ExpandString(" , ", ',') == StringListWrapper{{""}}); // {"", ""}}
  100. CHECK(ExpandString(" a b c ", ' ') == StringListWrapper{{"", "a", "", "b", "", "c"}}); // {"a", "b", "c"}}
  101. CHECK(ExpandStringShort("'a,b',c", ',') == StringListWrapper{{"a,b", "c"}}); // {"'a,b'", "c"}
  102. CHECK(ExpandStringShort(R"("a,b",c)", ',') == StringListWrapper{{R"(a,b)", "c"}}); // {R"("a,b")", "c"}
  103. CHECK(ExpandStringShort("\"a\\\",b\",c", ',') == StringListWrapper{{R"(a\",b)", "c"}}); // {R"("a\",b")", "c"} (MSVC fails using raw string here)
  104. CHECK(ExpandStringLong("\"a,b,c\",d", ',', '"', '"', false) == StringListWrapper{{"\"a,b,c\",d"}}); // {"\"a,b,c\"", "d"}}
  105. }
  106. TEST_CASE("StringUtilities::StartsWith")
  107. {
  108. using namespace Rml::StringUtilities;
  109. CHECK(StartsWith("abc", "abc"));
  110. CHECK(StartsWith("abc", "ab"));
  111. CHECK(StartsWith("abc", "a"));
  112. CHECK(StartsWith("abc", ""));
  113. CHECK(!StartsWith("abc", "abcd"));
  114. CHECK(!StartsWith("abc", "abd"));
  115. CHECK(!StartsWith("abc", "bbc"));
  116. CHECK(!StartsWith("abc", "bc"));
  117. CHECK(!StartsWith("abc", "x"));
  118. }
  119. TEST_CASE("StringUtilities::EndsWith")
  120. {
  121. using namespace Rml::StringUtilities;
  122. CHECK(EndsWith("abc", "abc"));
  123. CHECK(EndsWith("abc", "bc"));
  124. CHECK(EndsWith("abc", "c"));
  125. CHECK(EndsWith("abc", ""));
  126. CHECK(!EndsWith("abc", "abcd"));
  127. CHECK(!EndsWith("abc", "abd"));
  128. CHECK(!EndsWith("abc", "bbc"));
  129. CHECK(!EndsWith("abc", "ab"));
  130. CHECK(!EndsWith("abc", "x"));
  131. }
  132. TEST_CASE("StringView")
  133. {
  134. const char abc[] = "abc";
  135. const String str_abc = abc;
  136. CHECK(StringView("abc") == StringView("abc"));
  137. CHECK(StringView("abc") == String("abc"));
  138. CHECK(StringView("abc") == "abc");
  139. CHECK(StringView("abc") == abc);
  140. CHECK(StringView(String(abc)) == abc);
  141. CHECK(StringView(abc) == abc);
  142. CHECK(StringView(abc, abc + 3) == abc);
  143. CHECK(StringView(str_abc, 1) == "bc");
  144. CHECK(StringView(str_abc, 1, 1) == "b");
  145. CHECK(StringView("abcd") != abc);
  146. CHECK(StringView("ab") != abc);
  147. CHECK(StringView() != abc);
  148. CHECK(StringView() == String());
  149. CHECK(StringView() == "");
  150. }
  151. TEST_CASE("StringUtilities::ConvertByteOffsetToCharacterOffset")
  152. {
  153. using namespace Rml::StringUtilities;
  154. // clang-format off
  155. CHECK(ConvertByteOffsetToCharacterOffset("", 0) == 0);
  156. CHECK(ConvertByteOffsetToCharacterOffset("", 1) == 0);
  157. CHECK(ConvertByteOffsetToCharacterOffset("a", 0) == 0);
  158. CHECK(ConvertByteOffsetToCharacterOffset("a", 1) == 1);
  159. CHECK(ConvertByteOffsetToCharacterOffset("ab", 1) == 1);
  160. CHECK(ConvertByteOffsetToCharacterOffset("ab", 2) == 2);
  161. CHECK(ConvertByteOffsetToCharacterOffset("a\xC2\xA3" "b", 1) == 1);
  162. CHECK(ConvertByteOffsetToCharacterOffset("a\xC2\xA3" "b", 2) == 2);
  163. CHECK(ConvertByteOffsetToCharacterOffset("a\xC2\xA3" "b", 3) == 2);
  164. CHECK(ConvertByteOffsetToCharacterOffset("a\xC2\xA3" "b", 4) == 3);
  165. CHECK(ConvertByteOffsetToCharacterOffset("a\xE2\x82\xAC" "b", 2) == 2);
  166. CHECK(ConvertByteOffsetToCharacterOffset("a\xE2\x82\xAC" "b", 3) == 2);
  167. CHECK(ConvertByteOffsetToCharacterOffset("a\xE2\x82\xAC" "b", 4) == 2);
  168. CHECK(ConvertByteOffsetToCharacterOffset("a\xE2\x82\xAC" "b", 5) == 3);
  169. // clang-format on
  170. }
  171. TEST_CASE("StringUtilities::ConvertCharacterOffsetToByteOffset")
  172. {
  173. using namespace Rml::StringUtilities;
  174. // clang-format off
  175. CHECK(ConvertCharacterOffsetToByteOffset("", 0) == 0);
  176. CHECK(ConvertCharacterOffsetToByteOffset("", 1) == 0);
  177. CHECK(ConvertCharacterOffsetToByteOffset("a", 0) == 0);
  178. CHECK(ConvertCharacterOffsetToByteOffset("a", 1) == 1);
  179. CHECK(ConvertCharacterOffsetToByteOffset("ab", 1) == 1);
  180. CHECK(ConvertCharacterOffsetToByteOffset("ab", 2) == 2);
  181. CHECK(ConvertCharacterOffsetToByteOffset("a\xC2\xA3" "b", 1) == 1);
  182. CHECK(ConvertCharacterOffsetToByteOffset("a\xC2\xA3" "b", 2) == 3);
  183. CHECK(ConvertCharacterOffsetToByteOffset("a\xC2\xA3" "b", 3) == 4);
  184. CHECK(ConvertCharacterOffsetToByteOffset("a\xC2\xA3" "b", 4) == 4);
  185. CHECK(ConvertCharacterOffsetToByteOffset("a\xE2\x82\xAC" "b", 1) == 1);
  186. CHECK(ConvertCharacterOffsetToByteOffset("a\xE2\x82\xAC" "b", 2) == 4);
  187. CHECK(ConvertCharacterOffsetToByteOffset("a\xE2\x82\xAC" "b", 3) == 5);
  188. CHECK(ConvertCharacterOffsetToByteOffset("a\xE2\x82\xAC" "b", 4) == 5);
  189. // clang-format on
  190. }
  191. TEST_CASE("CreateString")
  192. {
  193. CHECK(Rml::CreateString("Hello %s!", "world") == "Hello world!");
  194. CHECK(Rml::CreateString("%g, %d, %.2f", 0.5f, 5, 2.f) == "0.5, 5, 2.00");
  195. constexpr int InternalBufferSize = 256;
  196. for (int string_size : {InternalBufferSize - 1, InternalBufferSize, InternalBufferSize + 1})
  197. {
  198. Rml::String large_string(string_size, 'x');
  199. CHECK(Rml::CreateString("%s", large_string.c_str()) == large_string);
  200. }
  201. }
  202. TEST_CASE("FormatString")
  203. {
  204. {
  205. Rml::String result;
  206. int length = Rml::FormatString(result, "Hello %s!", "world");
  207. CHECK(result == "Hello world!");
  208. CHECK(length == 12);
  209. }
  210. constexpr int InternalBufferSize = 256;
  211. for (int string_size : {InternalBufferSize - 1, InternalBufferSize, InternalBufferSize + 1})
  212. {
  213. const Rml::String large_string(string_size, 'x');
  214. Rml::String result;
  215. int length = Rml::FormatString(result, "%s", large_string.c_str());
  216. CHECK(result == large_string);
  217. CHECK(length == string_size);
  218. }
  219. }