Index: base/strings/string_util.cc |
diff --git a/base/strings/string_util.cc b/base/strings/string_util.cc |
index cc7769343b5a592a2d081c89e3203dc82be8f720..3317740d776a4f962efcb1a33ea2cbc8d06dda19 100644 |
--- a/base/strings/string_util.cc |
+++ b/base/strings/string_util.cc |
@@ -21,14 +21,13 @@ |
#include "base/basictypes.h" |
#include "base/logging.h" |
#include "base/memory/singleton.h" |
+#include "base/strings/string_split.h" |
#include "base/strings/utf_string_conversion_utils.h" |
#include "base/strings/utf_string_conversions.h" |
#include "base/third_party/icu/icu_utf.h" |
#include "build/build_config.h" |
-// Remove when this entire file is in the base namespace. |
-using base::char16; |
-using base::string16; |
+namespace base { |
namespace { |
@@ -79,14 +78,16 @@ template<typename T> inline T* AlignToMachineWord(T* pointer) { |
} |
template<size_t size, typename CharacterType> struct NonASCIIMask; |
-template<> struct NonASCIIMask<4, base::char16> { |
- static inline uint32_t value() { return 0xFF80FF80U; } |
+template <> |
+struct NonASCIIMask<4, char16> { |
+ static inline uint32_t value() { return 0xFF80FF80U; } |
}; |
template<> struct NonASCIIMask<4, char> { |
static inline uint32_t value() { return 0x80808080U; } |
}; |
-template<> struct NonASCIIMask<8, base::char16> { |
- static inline uint64_t value() { return 0xFF80FF80FF80FF80ULL; } |
+template <> |
+struct NonASCIIMask<8, char16> { |
+ static inline uint64_t value() { return 0xFF80FF80FF80FF80ULL; } |
}; |
template<> struct NonASCIIMask<8, char> { |
static inline uint64_t value() { return 0x8080808080808080ULL; } |
@@ -100,9 +101,17 @@ template<> struct NonASCIIMask<8, wchar_t> { |
}; |
#endif // WCHAR_T_IS_UTF32 |
-} // namespace |
+// DO NOT USE. http://crbug.com/24917 |
+// |
+// tolower() will given incorrect results for non-ASCII characters. Use the |
+// ASCII version, base::i18n::ToLower, or base::i18n::FoldCase. This is here |
+// for backwards-compat for StartsWith until such calls can be updated. |
+struct CaseInsensitiveCompareDeprecated { |
+ public: |
+ bool operator()(char16 x, char16 y) const { return tolower(x) == tolower(y); } |
+}; |
-namespace base { |
+} // namespace |
bool IsWprintfFormatPortable(const wchar_t* format) { |
for (const wchar_t* position = format; *position != '\0'; ++position) { |
@@ -139,6 +148,53 @@ bool IsWprintfFormatPortable(const wchar_t* format) { |
return true; |
} |
+template <class StringType> |
+int CompareCaseInsensitiveASCIIT(BasicStringPiece<StringType> a, |
+ BasicStringPiece<StringType> b) { |
+ // Find the first characters that aren't equal and compare them. If the end |
+ // of one of the strings is found before a nonequal character, the lengths |
+ // of the strings are compared. |
+ size_t i = 0; |
+ while (i < a.length() && i < b.length()) { |
+ typename StringType::value_type lower_a = ToLowerASCII(a[i]); |
+ typename StringType::value_type lower_b = ToLowerASCII(b[i]); |
+ if (lower_a < lower_b) |
+ return -1; |
+ if (lower_a > lower_b) |
+ return 1; |
+ i++; |
+ } |
+ |
+ // End of one string hit before finding a different character. Expect the |
+ // common case to be "strings equal" at this point so check that first. |
+ if (a.length() == b.length()) |
+ return 0; |
+ |
+ if (a.length() < b.length()) |
+ return -1; |
+ return 1; |
+} |
+ |
+int CompareCaseInsensitiveASCII(StringPiece a, StringPiece b) { |
+ return CompareCaseInsensitiveASCIIT<std::string>(a, b); |
+} |
+ |
+int CompareCaseInsensitiveASCII(StringPiece16 a, StringPiece16 b) { |
+ return CompareCaseInsensitiveASCIIT<string16>(a, b); |
+} |
+ |
+bool EqualsCaseInsensitiveASCII(StringPiece a, StringPiece b) { |
+ if (a.length() != b.length()) |
+ return false; |
+ return CompareCaseInsensitiveASCIIT<std::string>(a, b) == 0; |
+} |
+ |
+bool EqualsCaseInsensitiveASCII(StringPiece16 a, StringPiece16 b) { |
+ if (a.length() != b.length()) |
+ return false; |
+ return CompareCaseInsensitiveASCIIT<string16>(a, b) == 0; |
+} |
+ |
const std::string& EmptyString() { |
return EmptyStrings::GetInstance()->s; |
} |
@@ -168,54 +224,60 @@ bool ReplaceCharsT(const STR& input, |
} |
bool ReplaceChars(const string16& input, |
- const base::StringPiece16& replace_chars, |
+ const StringPiece16& replace_chars, |
const string16& replace_with, |
string16* output) { |
return ReplaceCharsT(input, replace_chars.as_string(), replace_with, output); |
} |
bool ReplaceChars(const std::string& input, |
- const base::StringPiece& replace_chars, |
+ const StringPiece& replace_chars, |
const std::string& replace_with, |
std::string* output) { |
return ReplaceCharsT(input, replace_chars.as_string(), replace_with, output); |
} |
bool RemoveChars(const string16& input, |
- const base::StringPiece16& remove_chars, |
+ const StringPiece16& remove_chars, |
string16* output) { |
return ReplaceChars(input, remove_chars.as_string(), string16(), output); |
} |
bool RemoveChars(const std::string& input, |
- const base::StringPiece& remove_chars, |
+ const StringPiece& remove_chars, |
std::string* output) { |
return ReplaceChars(input, remove_chars.as_string(), std::string(), output); |
} |
-template<typename STR> |
-TrimPositions TrimStringT(const STR& input, |
- const STR& trim_chars, |
+template <typename Str> |
+TrimPositions TrimStringT(const Str& input, |
+ BasicStringPiece<Str> trim_chars, |
TrimPositions positions, |
- STR* output) { |
- // Find the edges of leading/trailing whitespace as desired. |
+ Str* output) { |
+ // Find the edges of leading/trailing whitespace as desired. Need to use |
+ // a StringPiece version of input to be able to call find* on it with the |
+ // StringPiece version of trim_chars (normally the trim_chars will be a |
+ // constant so avoid making a copy). |
+ BasicStringPiece<Str> input_piece(input); |
const size_t last_char = input.length() - 1; |
- const size_t first_good_char = (positions & TRIM_LEADING) ? |
- input.find_first_not_of(trim_chars) : 0; |
- const size_t last_good_char = (positions & TRIM_TRAILING) ? |
- input.find_last_not_of(trim_chars) : last_char; |
- |
- // When the string was all whitespace, report that we stripped off whitespace |
- // from whichever position the caller was interested in. For empty input, we |
- // stripped no whitespace, but we still need to clear |output|. |
- if (input.empty() || |
- (first_good_char == STR::npos) || (last_good_char == STR::npos)) { |
+ const size_t first_good_char = (positions & TRIM_LEADING) |
+ ? input_piece.find_first_not_of(trim_chars) |
+ : 0; |
+ const size_t last_good_char = (positions & TRIM_TRAILING) |
+ ? input_piece.find_last_not_of(trim_chars) |
+ : last_char; |
+ |
+ // When the string was all trimmed, report that we stripped off characters |
+ // from whichever position the caller was interested in. For empty input, we |
+ // stripped no characters, but we still need to clear |output|. |
+ if (input.empty() || (first_good_char == Str::npos) || |
+ (last_good_char == Str::npos)) { |
bool input_was_empty = input.empty(); // in case output == &input |
output->clear(); |
return input_was_empty ? TRIM_NONE : positions; |
} |
- // Trim the whitespace. |
+ // Trim. |
*output = |
input.substr(first_good_char, last_good_char - first_good_char + 1); |
@@ -226,17 +288,39 @@ TrimPositions TrimStringT(const STR& input, |
} |
bool TrimString(const string16& input, |
- const base::StringPiece16& trim_chars, |
+ StringPiece16 trim_chars, |
string16* output) { |
- return TrimStringT(input, trim_chars.as_string(), TRIM_ALL, output) != |
- TRIM_NONE; |
+ return TrimStringT(input, trim_chars, TRIM_ALL, output) != TRIM_NONE; |
} |
bool TrimString(const std::string& input, |
- const base::StringPiece& trim_chars, |
+ StringPiece trim_chars, |
std::string* output) { |
- return TrimStringT(input, trim_chars.as_string(), TRIM_ALL, output) != |
- TRIM_NONE; |
+ return TrimStringT(input, trim_chars, TRIM_ALL, output) != TRIM_NONE; |
+} |
+ |
+template <typename Str> |
+BasicStringPiece<Str> TrimStringPieceT(BasicStringPiece<Str> input, |
+ BasicStringPiece<Str> trim_chars, |
+ TrimPositions positions) { |
+ size_t begin = |
+ (positions & TRIM_LEADING) ? input.find_first_not_of(trim_chars) : 0; |
+ size_t end = (positions & TRIM_TRAILING) |
+ ? input.find_last_not_of(trim_chars) + 1 |
+ : input.size(); |
+ return input.substr(begin, end - begin); |
+} |
+ |
+StringPiece16 TrimString(StringPiece16 input, |
+ const StringPiece16& trim_chars, |
+ TrimPositions positions) { |
+ return TrimStringPieceT(input, trim_chars, positions); |
+} |
+ |
+StringPiece TrimString(StringPiece input, |
+ const StringPiece& trim_chars, |
+ TrimPositions positions) { |
+ return TrimStringPieceT(input, trim_chars, positions); |
} |
void TruncateUTF8ToByteSize(const std::string& input, |
@@ -278,14 +362,22 @@ void TruncateUTF8ToByteSize(const std::string& input, |
TrimPositions TrimWhitespace(const string16& input, |
TrimPositions positions, |
string16* output) { |
- return TrimStringT(input, base::string16(kWhitespaceUTF16), positions, |
- output); |
+ return TrimStringT(input, StringPiece16(kWhitespaceUTF16), positions, output); |
+} |
+ |
+StringPiece16 TrimWhitespaceASCII(StringPiece16 input, |
+ TrimPositions positions) { |
+ return TrimStringPieceT(input, StringPiece16(kWhitespaceUTF16), positions); |
} |
TrimPositions TrimWhitespaceASCII(const std::string& input, |
TrimPositions positions, |
std::string* output) { |
- return TrimStringT(input, std::string(kWhitespaceASCII), positions, output); |
+ return TrimStringT(input, StringPiece(kWhitespaceASCII), positions, output); |
+} |
+ |
+StringPiece TrimWhitespaceASCII(StringPiece input, TrimPositions positions) { |
+ return TrimStringPieceT(input, StringPiece(kWhitespaceASCII), positions); |
} |
// This function is only for backward-compatibility. |
@@ -309,7 +401,7 @@ STR CollapseWhitespaceT(const STR& text, |
int chars_written = 0; |
for (typename STR::const_iterator i(text.begin()); i != text.end(); ++i) { |
- if (IsWhitespace(*i)) { |
+ if (IsUnicodeWhitespace(*i)) { |
if (!in_whitespace) { |
// Reduce all whitespace sequences to a single space. |
in_whitespace = true; |
@@ -482,55 +574,123 @@ bool EqualsASCII(const string16& a, const StringPiece& b) { |
return std::equal(b.begin(), b.end(), a.begin()); |
} |
-} // namespace base |
+template <typename Str> |
+bool StartsWithT(BasicStringPiece<Str> str, |
+ BasicStringPiece<Str> search_for, |
+ CompareCase case_sensitivity) { |
+ if (search_for.size() > str.size()) |
+ return false; |
-bool StartsWithASCII(const std::string& str, |
- const std::string& search, |
- bool case_sensitive) { |
- if (case_sensitive) |
- return str.compare(0, search.length(), search) == 0; |
- else |
- return base::strncasecmp(str.c_str(), search.c_str(), search.length()) == 0; |
+ BasicStringPiece<Str> source = str.substr(0, search_for.size()); |
+ |
+ switch (case_sensitivity) { |
+ case CompareCase::SENSITIVE: |
+ return source == search_for; |
+ |
+ case CompareCase::INSENSITIVE_ASCII: |
+ return std::equal( |
+ search_for.begin(), search_for.end(), source.begin(), |
+ CaseInsensitiveCompareASCII<typename Str::value_type>()); |
+ |
+ default: |
+ NOTREACHED(); |
+ return false; |
+ } |
} |
-template <typename STR> |
-bool StartsWithT(const STR& str, const STR& search, bool case_sensitive) { |
- if (case_sensitive) { |
- return str.compare(0, search.length(), search) == 0; |
- } else { |
+bool StartsWith(StringPiece str, |
+ StringPiece search_for, |
+ CompareCase case_sensitivity) { |
+ return StartsWithT<std::string>(str, search_for, case_sensitivity); |
+} |
+ |
+bool StartsWith(StringPiece16 str, |
+ StringPiece16 search_for, |
+ CompareCase case_sensitivity) { |
+ return StartsWithT<string16>(str, search_for, case_sensitivity); |
+} |
+ |
+bool StartsWith(const string16& str, |
+ const string16& search, |
+ bool case_sensitive) { |
+ if (!case_sensitive) { |
+ // This function was originally written using the current locale functions |
+ // for case-insensitive comparisons. Emulate this behavior until callers |
+ // can be converted either to use the case-insensitive ASCII one (most |
+ // callers) or ICU functions in base_i18n. |
if (search.size() > str.size()) |
return false; |
return std::equal(search.begin(), search.end(), str.begin(), |
- base::CaseInsensitiveCompare<typename STR::value_type>()); |
+ CaseInsensitiveCompareDeprecated()); |
} |
+ return StartsWith(StringPiece16(str), StringPiece16(search), |
+ CompareCase::SENSITIVE); |
} |
-bool StartsWith(const string16& str, const string16& search, |
- bool case_sensitive) { |
- return StartsWithT(str, search, case_sensitive); |
+template <typename Str> |
+bool EndsWithT(BasicStringPiece<Str> str, |
+ BasicStringPiece<Str> search_for, |
+ CompareCase case_sensitivity) { |
+ if (search_for.size() > str.size()) |
+ return false; |
+ |
+ BasicStringPiece<Str> source = |
+ str.substr(str.size() - search_for.size(), search_for.size()); |
+ |
+ switch (case_sensitivity) { |
+ case CompareCase::SENSITIVE: |
+ return source == search_for; |
+ |
+ case CompareCase::INSENSITIVE_ASCII: |
+ return std::equal( |
+ source.begin(), source.end(), search_for.begin(), |
+ CaseInsensitiveCompareASCII<typename Str::value_type>()); |
+ |
+ default: |
+ NOTREACHED(); |
+ return false; |
+ } |
} |
-template <typename STR> |
-bool EndsWithT(const STR& str, const STR& search, bool case_sensitive) { |
- size_t str_length = str.length(); |
- size_t search_length = search.length(); |
- if (search_length > str_length) |
- return false; |
- if (case_sensitive) |
- return str.compare(str_length - search_length, search_length, search) == 0; |
- return std::equal(search.begin(), search.end(), |
- str.begin() + (str_length - search_length), |
- base::CaseInsensitiveCompare<typename STR::value_type>()); |
+bool EndsWith(StringPiece str, |
+ StringPiece search_for, |
+ CompareCase case_sensitivity) { |
+ return EndsWithT<std::string>(str, search_for, case_sensitivity); |
} |
-bool EndsWith(const std::string& str, const std::string& search, |
- bool case_sensitive) { |
- return EndsWithT(str, search, case_sensitive); |
+bool EndsWith(StringPiece16 str, |
+ StringPiece16 search_for, |
+ CompareCase case_sensitivity) { |
+ return EndsWithT<string16>(str, search_for, case_sensitivity); |
} |
-bool EndsWith(const string16& str, const string16& search, |
+bool EndsWith(const string16& str, |
+ const string16& search, |
bool case_sensitive) { |
- return EndsWithT(str, search, case_sensitive); |
+ if (!case_sensitive) { |
+ // This function was originally written using the current locale functions |
+ // for case-insensitive comparisons. Emulate this behavior until callers |
+ // can be converted either to use the case-insensitive ASCII one (most |
+ // callers) or ICU functions in base_i18n. |
+ if (search.size() > str.size()) |
+ return false; |
+ return std::equal(search.begin(), search.end(), |
+ str.begin() + (str.size() - search.size()), |
+ CaseInsensitiveCompareDeprecated()); |
+ } |
+ return EndsWith(StringPiece16(str), StringPiece16(search), |
+ CompareCase::SENSITIVE); |
+} |
+ |
+char HexDigitToInt(wchar_t c) { |
+ DCHECK(IsHexDigit(c)); |
+ if (c >= '0' && c <= '9') |
+ return static_cast<char>(c - '0'); |
+ if (c >= 'A' && c <= 'F') |
+ return static_cast<char>(c - 'A' + 10); |
+ if (c >= 'a' && c <= 'f') |
+ return static_cast<char>(c - 'a' + 10); |
+ return 0; |
} |
static const char* const kByteStringsUnlocalized[] = { |
@@ -561,20 +721,20 @@ string16 FormatBytesUnlocalized(int64 bytes) { |
kByteStringsUnlocalized[dimension]); |
} |
- return base::ASCIIToUTF16(buf); |
+ return ASCIIToUTF16(buf); |
} |
// Runs in O(n) time in the length of |str|. |
-template<class StringType> |
+template <class StringType> |
void DoReplaceSubstringsAfterOffset(StringType* str, |
size_t offset, |
- const StringType& find_this, |
- const StringType& replace_with, |
+ BasicStringPiece<StringType> find_this, |
+ BasicStringPiece<StringType> replace_with, |
bool replace_all) { |
DCHECK(!find_this.empty()); |
// If the find string doesn't appear, there's nothing to do. |
- offset = str->find(find_this, offset); |
+ offset = str->find(find_this.data(), offset, find_this.size()); |
if (offset == StringType::npos) |
return; |
@@ -582,7 +742,7 @@ void DoReplaceSubstringsAfterOffset(StringType* str, |
// complicated. |
size_t find_length = find_this.length(); |
if (!replace_all) { |
- str->replace(offset, find_length, replace_with); |
+ str->replace(offset, find_length, replace_with.data(), replace_with.size()); |
return; |
} |
@@ -591,8 +751,10 @@ void DoReplaceSubstringsAfterOffset(StringType* str, |
size_t replace_length = replace_with.length(); |
if (find_length == replace_length) { |
do { |
- str->replace(offset, find_length, replace_with); |
- offset = str->find(find_this, offset + replace_length); |
+ str->replace(offset, find_length, replace_with.data(), |
+ replace_with.size()); |
+ offset = str->find(find_this.data(), offset + replace_length, |
+ find_this.size()); |
} while (offset != StringType::npos); |
return; |
} |
@@ -609,11 +771,14 @@ void DoReplaceSubstringsAfterOffset(StringType* str, |
size_t write_offset = offset; |
do { |
if (replace_length) { |
- str->replace(write_offset, replace_length, replace_with); |
+ str->replace(write_offset, replace_length, replace_with.data(), |
+ replace_with.size()); |
write_offset += replace_length; |
} |
size_t read_offset = offset + find_length; |
- offset = std::min(str->find(find_this, read_offset), str_length); |
+ offset = |
+ std::min(str->find(find_this.data(), read_offset, find_this.size()), |
+ str_length); |
size_t length = offset - read_offset; |
if (length) { |
memmove(&(*str)[write_offset], &(*str)[read_offset], |
@@ -642,13 +807,15 @@ void DoReplaceSubstringsAfterOffset(StringType* str, |
// exit from the loop, |current_match| will point at the last instance of |
// the find string, and we won't need to find() it again immediately. |
current_match = offset; |
- offset = str->find(find_this, offset + find_length); |
+ offset = |
+ str->find(find_this.data(), offset + find_length, find_this.size()); |
} while (offset != StringType::npos); |
str->resize(final_length); |
// Now do the replacement loop, working backwards through the string. |
- for (size_t prev_match = str_length, write_offset = final_length; ; |
- current_match = str->rfind(find_this, current_match - 1)) { |
+ for (size_t prev_match = str_length, write_offset = final_length;; |
+ current_match = |
+ str->rfind(find_this.data(), current_match - 1, find_this.size())) { |
size_t read_offset = current_match + find_length; |
size_t length = prev_match - read_offset; |
if (length) { |
@@ -657,7 +824,8 @@ void DoReplaceSubstringsAfterOffset(StringType* str, |
length * sizeof(typename StringType::value_type)); |
} |
write_offset -= replace_length; |
- str->replace(write_offset, replace_length, replace_with); |
+ str->replace(write_offset, replace_length, replace_with.data(), |
+ replace_with.size()); |
if (current_match == first_match) |
return; |
prev_match = current_match; |
@@ -666,128 +834,97 @@ void DoReplaceSubstringsAfterOffset(StringType* str, |
void ReplaceFirstSubstringAfterOffset(string16* str, |
size_t start_offset, |
- const string16& find_this, |
- const string16& replace_with) { |
- DoReplaceSubstringsAfterOffset(str, start_offset, find_this, replace_with, |
- false); // replace first instance |
+ StringPiece16 find_this, |
+ StringPiece16 replace_with) { |
+ DoReplaceSubstringsAfterOffset<string16>( |
+ str, start_offset, find_this, replace_with, false); // Replace first. |
} |
void ReplaceFirstSubstringAfterOffset(std::string* str, |
size_t start_offset, |
- const std::string& find_this, |
- const std::string& replace_with) { |
- DoReplaceSubstringsAfterOffset(str, start_offset, find_this, replace_with, |
- false); // replace first instance |
+ StringPiece find_this, |
+ StringPiece replace_with) { |
+ DoReplaceSubstringsAfterOffset<std::string>( |
+ str, start_offset, find_this, replace_with, false); // Replace first. |
} |
void ReplaceSubstringsAfterOffset(string16* str, |
size_t start_offset, |
- const string16& find_this, |
- const string16& replace_with) { |
- DoReplaceSubstringsAfterOffset(str, start_offset, find_this, replace_with, |
- true); // replace all instances |
+ StringPiece16 find_this, |
+ StringPiece16 replace_with) { |
+ DoReplaceSubstringsAfterOffset<string16>(str, start_offset, find_this, |
+ replace_with, true); // Replace all. |
} |
void ReplaceSubstringsAfterOffset(std::string* str, |
size_t start_offset, |
- const std::string& find_this, |
- const std::string& replace_with) { |
- DoReplaceSubstringsAfterOffset(str, start_offset, find_this, replace_with, |
- true); // replace all instances |
+ StringPiece find_this, |
+ StringPiece replace_with) { |
+ DoReplaceSubstringsAfterOffset<std::string>( |
+ str, start_offset, find_this, replace_with, true); // Replace all. |
} |
- |
-template<typename STR> |
-static size_t TokenizeT(const STR& str, |
- const STR& delimiters, |
- std::vector<STR>* tokens) { |
- tokens->clear(); |
- |
- size_t start = str.find_first_not_of(delimiters); |
- while (start != STR::npos) { |
- size_t end = str.find_first_of(delimiters, start + 1); |
- if (end == STR::npos) { |
- tokens->push_back(str.substr(start)); |
- break; |
- } else { |
- tokens->push_back(str.substr(start, end - start)); |
- start = str.find_first_not_of(delimiters, end + 1); |
- } |
- } |
- |
- return tokens->size(); |
-} |
- |
-size_t Tokenize(const string16& str, |
- const string16& delimiters, |
- std::vector<string16>* tokens) { |
- return TokenizeT(str, delimiters, tokens); |
+template <class string_type> |
+inline typename string_type::value_type* WriteIntoT(string_type* str, |
+ size_t length_with_null) { |
+ DCHECK_GT(length_with_null, 1u); |
+ str->reserve(length_with_null); |
+ str->resize(length_with_null - 1); |
+ return &((*str)[0]); |
} |
-size_t Tokenize(const std::string& str, |
- const std::string& delimiters, |
- std::vector<std::string>* tokens) { |
- return TokenizeT(str, delimiters, tokens); |
+char* WriteInto(std::string* str, size_t length_with_null) { |
+ return WriteIntoT(str, length_with_null); |
} |
-size_t Tokenize(const base::StringPiece& str, |
- const base::StringPiece& delimiters, |
- std::vector<base::StringPiece>* tokens) { |
- return TokenizeT(str, delimiters, tokens); |
+char16* WriteInto(string16* str, size_t length_with_null) { |
+ return WriteIntoT(str, length_with_null); |
} |
-template<typename STR> |
-static STR JoinStringT(const std::vector<STR>& parts, const STR& sep) { |
+template <typename STR> |
+static STR JoinStringT(const std::vector<STR>& parts, |
+ BasicStringPiece<STR> sep) { |
if (parts.empty()) |
return STR(); |
STR result(parts[0]); |
- typename std::vector<STR>::const_iterator iter = parts.begin(); |
+ auto iter = parts.begin(); |
++iter; |
for (; iter != parts.end(); ++iter) { |
- result += sep; |
+ sep.AppendToString(&result); |
result += *iter; |
} |
return result; |
} |
-std::string JoinString(const std::vector<std::string>& parts, char sep) { |
- return JoinStringT(parts, std::string(1, sep)); |
-} |
- |
-string16 JoinString(const std::vector<string16>& parts, char16 sep) { |
- return JoinStringT(parts, string16(1, sep)); |
-} |
- |
std::string JoinString(const std::vector<std::string>& parts, |
- const std::string& separator) { |
+ StringPiece separator) { |
return JoinStringT(parts, separator); |
} |
string16 JoinString(const std::vector<string16>& parts, |
- const string16& separator) { |
+ StringPiece16 separator) { |
return JoinStringT(parts, separator); |
} |
-template<class FormatStringType, class OutStringType> |
-OutStringType DoReplaceStringPlaceholders(const FormatStringType& format_string, |
- const std::vector<OutStringType>& subst, std::vector<size_t>* offsets) { |
+template <class FormatStringType, class OutStringType> |
+OutStringType DoReplaceStringPlaceholders( |
+ const FormatStringType& format_string, |
+ const std::vector<OutStringType>& subst, |
+ std::vector<size_t>* offsets) { |
size_t substitutions = subst.size(); |
size_t sub_length = 0; |
- for (typename std::vector<OutStringType>::const_iterator iter = subst.begin(); |
- iter != subst.end(); ++iter) { |
- sub_length += iter->length(); |
- } |
+ for (const auto& cur : subst) |
+ sub_length += cur.length(); |
OutStringType formatted; |
formatted.reserve(format_string.length() + sub_length); |
std::vector<ReplacementOffset> r_offsets; |
- for (typename FormatStringType::const_iterator i = format_string.begin(); |
- i != format_string.end(); ++i) { |
+ for (auto i = format_string.begin(); i != format_string.end(); ++i) { |
if ('$' == *i) { |
if (i + 1 != format_string.end()) { |
++i; |
@@ -825,10 +962,8 @@ OutStringType DoReplaceStringPlaceholders(const FormatStringType& format_string, |
} |
} |
if (offsets) { |
- for (std::vector<ReplacementOffset>::const_iterator i = r_offsets.begin(); |
- i != r_offsets.end(); ++i) { |
- offsets->push_back(i->offset); |
- } |
+ for (const auto& cur : r_offsets) |
+ offsets->push_back(cur.offset); |
} |
return formatted; |
} |
@@ -839,7 +974,7 @@ string16 ReplaceStringPlaceholders(const string16& format_string, |
return DoReplaceStringPlaceholders(format_string, subst, offsets); |
} |
-std::string ReplaceStringPlaceholders(const base::StringPiece& format_string, |
+std::string ReplaceStringPlaceholders(const StringPiece& format_string, |
const std::vector<std::string>& subst, |
std::vector<size_t>* offsets) { |
return DoReplaceStringPlaceholders(format_string, subst, offsets); |
@@ -859,161 +994,6 @@ string16 ReplaceStringPlaceholders(const string16& format_string, |
return result; |
} |
-static bool IsWildcard(base_icu::UChar32 character) { |
- return character == '*' || character == '?'; |
-} |
- |
-// Move the strings pointers to the point where they start to differ. |
-template <typename CHAR, typename NEXT> |
-static void EatSameChars(const CHAR** pattern, const CHAR* pattern_end, |
- const CHAR** string, const CHAR* string_end, |
- NEXT next) { |
- const CHAR* escape = NULL; |
- while (*pattern != pattern_end && *string != string_end) { |
- if (!escape && IsWildcard(**pattern)) { |
- // We don't want to match wildcard here, except if it's escaped. |
- return; |
- } |
- |
- // Check if the escapement char is found. If so, skip it and move to the |
- // next character. |
- if (!escape && **pattern == '\\') { |
- escape = *pattern; |
- next(pattern, pattern_end); |
- continue; |
- } |
- |
- // Check if the chars match, if so, increment the ptrs. |
- const CHAR* pattern_next = *pattern; |
- const CHAR* string_next = *string; |
- base_icu::UChar32 pattern_char = next(&pattern_next, pattern_end); |
- if (pattern_char == next(&string_next, string_end) && |
- pattern_char != CBU_SENTINEL) { |
- *pattern = pattern_next; |
- *string = string_next; |
- } else { |
- // Uh oh, it did not match, we are done. If the last char was an |
- // escapement, that means that it was an error to advance the ptr here, |
- // let's put it back where it was. This also mean that the MatchPattern |
- // function will return false because if we can't match an escape char |
- // here, then no one will. |
- if (escape) { |
- *pattern = escape; |
- } |
- return; |
- } |
- |
- escape = NULL; |
- } |
-} |
- |
-template <typename CHAR, typename NEXT> |
-static void EatWildcard(const CHAR** pattern, const CHAR* end, NEXT next) { |
- while (*pattern != end) { |
- if (!IsWildcard(**pattern)) |
- return; |
- next(pattern, end); |
- } |
-} |
- |
-template <typename CHAR, typename NEXT> |
-static bool MatchPatternT(const CHAR* eval, const CHAR* eval_end, |
- const CHAR* pattern, const CHAR* pattern_end, |
- int depth, |
- NEXT next) { |
- const int kMaxDepth = 16; |
- if (depth > kMaxDepth) |
- return false; |
- |
- // Eat all the matching chars. |
- EatSameChars(&pattern, pattern_end, &eval, eval_end, next); |
- |
- // If the string is empty, then the pattern must be empty too, or contains |
- // only wildcards. |
- if (eval == eval_end) { |
- EatWildcard(&pattern, pattern_end, next); |
- return pattern == pattern_end; |
- } |
- |
- // Pattern is empty but not string, this is not a match. |
- if (pattern == pattern_end) |
- return false; |
- |
- // If this is a question mark, then we need to compare the rest with |
- // the current string or the string with one character eaten. |
- const CHAR* next_pattern = pattern; |
- next(&next_pattern, pattern_end); |
- if (pattern[0] == '?') { |
- if (MatchPatternT(eval, eval_end, next_pattern, pattern_end, |
- depth + 1, next)) |
- return true; |
- const CHAR* next_eval = eval; |
- next(&next_eval, eval_end); |
- if (MatchPatternT(next_eval, eval_end, next_pattern, pattern_end, |
- depth + 1, next)) |
- return true; |
- } |
- |
- // This is a *, try to match all the possible substrings with the remainder |
- // of the pattern. |
- if (pattern[0] == '*') { |
- // Collapse duplicate wild cards (********** into *) so that the |
- // method does not recurse unnecessarily. http://crbug.com/52839 |
- EatWildcard(&next_pattern, pattern_end, next); |
- |
- while (eval != eval_end) { |
- if (MatchPatternT(eval, eval_end, next_pattern, pattern_end, |
- depth + 1, next)) |
- return true; |
- eval++; |
- } |
- |
- // We reached the end of the string, let see if the pattern contains only |
- // wildcards. |
- if (eval == eval_end) { |
- EatWildcard(&pattern, pattern_end, next); |
- if (pattern != pattern_end) |
- return false; |
- return true; |
- } |
- } |
- |
- return false; |
-} |
- |
-struct NextCharUTF8 { |
- base_icu::UChar32 operator()(const char** p, const char* end) { |
- base_icu::UChar32 c; |
- int offset = 0; |
- CBU8_NEXT(*p, offset, end - *p, c); |
- *p += offset; |
- return c; |
- } |
-}; |
- |
-struct NextCharUTF16 { |
- base_icu::UChar32 operator()(const char16** p, const char16* end) { |
- base_icu::UChar32 c; |
- int offset = 0; |
- CBU16_NEXT(*p, offset, end - *p, c); |
- *p += offset; |
- return c; |
- } |
-}; |
- |
-bool MatchPattern(const base::StringPiece& eval, |
- const base::StringPiece& pattern) { |
- return MatchPatternT(eval.data(), eval.data() + eval.size(), |
- pattern.data(), pattern.data() + pattern.size(), |
- 0, NextCharUTF8()); |
-} |
- |
-bool MatchPattern(const string16& eval, const string16& pattern) { |
- return MatchPatternT(eval.c_str(), eval.c_str() + eval.size(), |
- pattern.c_str(), pattern.c_str() + pattern.size(), |
- 0, NextCharUTF16()); |
-} |
- |
// The following code is compatible with the OpenBSD lcpy interface. See: |
// http://www.gratisoft.us/todd/papers/strlcpy.html |
// ftp://ftp.openbsd.org/pub/OpenBSD/src/lib/libc/string/{wcs,str}lcpy.c |
@@ -1038,9 +1018,11 @@ size_t lcpyT(CHAR* dst, const CHAR* src, size_t dst_size) { |
} // namespace |
-size_t base::strlcpy(char* dst, const char* src, size_t dst_size) { |
+size_t strlcpy(char* dst, const char* src, size_t dst_size) { |
return lcpyT<char>(dst, src, dst_size); |
} |
-size_t base::wcslcpy(wchar_t* dst, const wchar_t* src, size_t dst_size) { |
+size_t wcslcpy(wchar_t* dst, const wchar_t* src, size_t dst_size) { |
return lcpyT<wchar_t>(dst, src, dst_size); |
} |
+ |
+} // namespace base |