StringUtilities.cpp 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199
  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 <RmlUi/Core/StringUtilities.h>
  29. #include <RmlUi/Core/Types.h>
  30. #include <doctest.h>
  31. using namespace Rml;
  32. TEST_CASE("StringUtilities::TrimTrailingDotZeros")
  33. {
  34. auto RunTrimTrailingDotZeros = [](String string) {
  35. StringUtilities::TrimTrailingDotZeros(string);
  36. return string;
  37. };
  38. CHECK(RunTrimTrailingDotZeros("0.1") == "0.1");
  39. CHECK(RunTrimTrailingDotZeros("0.10") == "0.1");
  40. CHECK(RunTrimTrailingDotZeros("0.1000") == "0.1");
  41. CHECK(RunTrimTrailingDotZeros("0.01") == "0.01");
  42. CHECK(RunTrimTrailingDotZeros("0.") == "0");
  43. CHECK(RunTrimTrailingDotZeros("5.") == "5");
  44. CHECK(RunTrimTrailingDotZeros("5.5") == "5.5");
  45. CHECK(RunTrimTrailingDotZeros("5.50") == "5.5");
  46. CHECK(RunTrimTrailingDotZeros("5.501") == "5.501");
  47. CHECK(RunTrimTrailingDotZeros("10.0") == "10");
  48. CHECK(RunTrimTrailingDotZeros("11.0") == "11");
  49. // Some test cases for behavior that are probably not what you want.
  50. WARN(RunTrimTrailingDotZeros("test0") == "test");
  51. WARN(RunTrimTrailingDotZeros("1000") == "1");
  52. WARN(RunTrimTrailingDotZeros(".") == "");
  53. WARN(RunTrimTrailingDotZeros("0") == "");
  54. WARN(RunTrimTrailingDotZeros(".0") == "");
  55. WARN(RunTrimTrailingDotZeros(" 11 2121 3.00") == " 11 2121 3");
  56. WARN(RunTrimTrailingDotZeros("11") == "11");
  57. }
  58. TEST_CASE("StringUtilities::StartsWith")
  59. {
  60. using namespace Rml::StringUtilities;
  61. CHECK(StartsWith("abc", "abc"));
  62. CHECK(StartsWith("abc", "ab"));
  63. CHECK(StartsWith("abc", "a"));
  64. CHECK(StartsWith("abc", ""));
  65. CHECK(!StartsWith("abc", "abcd"));
  66. CHECK(!StartsWith("abc", "abd"));
  67. CHECK(!StartsWith("abc", "bbc"));
  68. CHECK(!StartsWith("abc", "bc"));
  69. CHECK(!StartsWith("abc", "x"));
  70. }
  71. TEST_CASE("StringUtilities::EndsWith")
  72. {
  73. using namespace Rml::StringUtilities;
  74. CHECK(EndsWith("abc", "abc"));
  75. CHECK(EndsWith("abc", "bc"));
  76. CHECK(EndsWith("abc", "c"));
  77. CHECK(EndsWith("abc", ""));
  78. CHECK(!EndsWith("abc", "abcd"));
  79. CHECK(!EndsWith("abc", "abd"));
  80. CHECK(!EndsWith("abc", "bbc"));
  81. CHECK(!EndsWith("abc", "ab"));
  82. CHECK(!EndsWith("abc", "x"));
  83. }
  84. TEST_CASE("StringView")
  85. {
  86. const char abc[] = "abc";
  87. const String str_abc = abc;
  88. CHECK(StringView("abc") == StringView("abc"));
  89. CHECK(StringView("abc") == String("abc"));
  90. CHECK(StringView("abc") == "abc");
  91. CHECK(StringView("abc") == abc);
  92. CHECK(StringView(String(abc)) == abc);
  93. CHECK(StringView(abc) == abc);
  94. CHECK(StringView(abc, abc + 3) == abc);
  95. CHECK(StringView(str_abc, 1) == "bc");
  96. CHECK(StringView(str_abc, 1, 1) == "b");
  97. CHECK(StringView("abcd") != abc);
  98. CHECK(StringView("ab") != abc);
  99. CHECK(StringView() != abc);
  100. CHECK(StringView() == String());
  101. CHECK(StringView() == "");
  102. }
  103. TEST_CASE("StringUtilities::ConvertByteOffsetToCharacterOffset")
  104. {
  105. using namespace Rml::StringUtilities;
  106. // clang-format off
  107. CHECK(ConvertByteOffsetToCharacterOffset("", 0) == 0);
  108. CHECK(ConvertByteOffsetToCharacterOffset("", 1) == 0);
  109. CHECK(ConvertByteOffsetToCharacterOffset("a", 0) == 0);
  110. CHECK(ConvertByteOffsetToCharacterOffset("a", 1) == 1);
  111. CHECK(ConvertByteOffsetToCharacterOffset("ab", 1) == 1);
  112. CHECK(ConvertByteOffsetToCharacterOffset("ab", 2) == 2);
  113. CHECK(ConvertByteOffsetToCharacterOffset("a\xC2\xA3" "b", 1) == 1);
  114. CHECK(ConvertByteOffsetToCharacterOffset("a\xC2\xA3" "b", 2) == 2);
  115. CHECK(ConvertByteOffsetToCharacterOffset("a\xC2\xA3" "b", 3) == 2);
  116. CHECK(ConvertByteOffsetToCharacterOffset("a\xC2\xA3" "b", 4) == 3);
  117. CHECK(ConvertByteOffsetToCharacterOffset("a\xE2\x82\xAC" "b", 2) == 2);
  118. CHECK(ConvertByteOffsetToCharacterOffset("a\xE2\x82\xAC" "b", 3) == 2);
  119. CHECK(ConvertByteOffsetToCharacterOffset("a\xE2\x82\xAC" "b", 4) == 2);
  120. CHECK(ConvertByteOffsetToCharacterOffset("a\xE2\x82\xAC" "b", 5) == 3);
  121. // clang-format on
  122. }
  123. TEST_CASE("StringUtilities::ConvertCharacterOffsetToByteOffset")
  124. {
  125. using namespace Rml::StringUtilities;
  126. // clang-format off
  127. CHECK(ConvertCharacterOffsetToByteOffset("", 0) == 0);
  128. CHECK(ConvertCharacterOffsetToByteOffset("", 1) == 0);
  129. CHECK(ConvertCharacterOffsetToByteOffset("a", 0) == 0);
  130. CHECK(ConvertCharacterOffsetToByteOffset("a", 1) == 1);
  131. CHECK(ConvertCharacterOffsetToByteOffset("ab", 1) == 1);
  132. CHECK(ConvertCharacterOffsetToByteOffset("ab", 2) == 2);
  133. CHECK(ConvertCharacterOffsetToByteOffset("a\xC2\xA3" "b", 1) == 1);
  134. CHECK(ConvertCharacterOffsetToByteOffset("a\xC2\xA3" "b", 2) == 3);
  135. CHECK(ConvertCharacterOffsetToByteOffset("a\xC2\xA3" "b", 3) == 4);
  136. CHECK(ConvertCharacterOffsetToByteOffset("a\xC2\xA3" "b", 4) == 4);
  137. CHECK(ConvertCharacterOffsetToByteOffset("a\xE2\x82\xAC" "b", 1) == 1);
  138. CHECK(ConvertCharacterOffsetToByteOffset("a\xE2\x82\xAC" "b", 2) == 4);
  139. CHECK(ConvertCharacterOffsetToByteOffset("a\xE2\x82\xAC" "b", 3) == 5);
  140. CHECK(ConvertCharacterOffsetToByteOffset("a\xE2\x82\xAC" "b", 4) == 5);
  141. // clang-format on
  142. }
  143. TEST_CASE("CreateString")
  144. {
  145. CHECK(Rml::CreateString("Hello %s!", "world") == "Hello world!");
  146. CHECK(Rml::CreateString("%g, %d, %.2f", 0.5f, 5, 2.f) == "0.5, 5, 2.00");
  147. constexpr int InternalBufferSize = 256;
  148. for (int string_size : {InternalBufferSize - 1, InternalBufferSize, InternalBufferSize + 1})
  149. {
  150. Rml::String large_string(string_size, 'x');
  151. CHECK(Rml::CreateString("%s", large_string.c_str()) == large_string);
  152. }
  153. }
  154. TEST_CASE("FormatString")
  155. {
  156. {
  157. Rml::String result;
  158. int length = Rml::FormatString(result, "Hello %s!", "world");
  159. CHECK(result == "Hello world!");
  160. CHECK(length == 12);
  161. }
  162. constexpr int InternalBufferSize = 256;
  163. for (int string_size : {InternalBufferSize - 1, InternalBufferSize, InternalBufferSize + 1})
  164. {
  165. const Rml::String large_string(string_size, 'x');
  166. Rml::String result;
  167. int length = Rml::FormatString(result, "%s", large_string.c_str());
  168. CHECK(result == large_string);
  169. CHECK(length == string_size);
  170. }
  171. }