| // Copyright (c) 2011 The Chromium Authors. All rights reserved. | 
 | // Use of this source code is governed by a BSD-style license that can be | 
 | // found in the LICENSE file. | 
 |  | 
 | #include "base/i18n/case_conversion.h" | 
 |  | 
 | #include <stdint.h> | 
 |  | 
 | #include "base/numerics/safe_conversions.h" | 
 | #include "base/strings/string16.h" | 
 | #include "base/strings/string_util.h" | 
 | #include "third_party/icu/source/common/unicode/uchar.h" | 
 | #include "third_party/icu/source/common/unicode/unistr.h" | 
 | #include "third_party/icu/source/common/unicode/ustring.h" | 
 |  | 
 | namespace base { | 
 | namespace i18n { | 
 |  | 
 | namespace { | 
 |  | 
 | // Provides a uniform interface for upper/lower/folding which take take | 
 | // slightly varying parameters. | 
 | typedef int32_t (*CaseMapperFunction)(UChar* dest, int32_t dest_capacity, | 
 |                                       const UChar* src, int32_t src_length, | 
 |                                       UErrorCode* error); | 
 |  | 
 | int32_t ToUpperMapper(UChar* dest, int32_t dest_capacity, | 
 |                       const UChar* src, int32_t src_length, | 
 |                       UErrorCode* error) { | 
 |   // Use default locale. | 
 |   return u_strToUpper(dest, dest_capacity, src, src_length, nullptr, error); | 
 | } | 
 |  | 
 | int32_t ToLowerMapper(UChar* dest, int32_t dest_capacity, | 
 |                       const UChar* src, int32_t src_length, | 
 |                       UErrorCode* error) { | 
 |   // Use default locale. | 
 |   return u_strToLower(dest, dest_capacity, src, src_length, nullptr, error); | 
 | } | 
 |  | 
 | int32_t FoldCaseMapper(UChar* dest, int32_t dest_capacity, | 
 |                        const UChar* src, int32_t src_length, | 
 |                        UErrorCode* error) { | 
 |   return u_strFoldCase(dest, dest_capacity, src, src_length, | 
 |                        U_FOLD_CASE_DEFAULT, error); | 
 | } | 
 |  | 
 | // Provides similar functionality as UnicodeString::caseMap but on string16. | 
 | string16 CaseMap(StringPiece16 string, CaseMapperFunction case_mapper) { | 
 |   string16 dest; | 
 |   if (string.empty()) | 
 |     return dest; | 
 |  | 
 |   // Provide an initial guess that the string length won't change. The typical | 
 |   // strings we use will very rarely change length in this process, so don't | 
 |   // optimize for that case. | 
 |   dest.resize(string.size()); | 
 |  | 
 |   UErrorCode error; | 
 |   do { | 
 |     error = U_ZERO_ERROR; | 
 |  | 
 |     // ICU won't terminate the string if there's not enough room for the null | 
 |     // terminator, but will otherwise. So we don't need to save room for that. | 
 |     // Don't use WriteInto, which assumes null terminators. | 
 |     int32_t new_length = case_mapper( | 
 |         &dest[0], saturated_cast<int32_t>(dest.size()), | 
 |         string.data(), saturated_cast<int32_t>(string.size()), | 
 |         &error); | 
 |     dest.resize(new_length); | 
 |   } while (error == U_BUFFER_OVERFLOW_ERROR); | 
 |   return dest; | 
 | } | 
 |  | 
 | }  // namespace | 
 |  | 
 | string16 ToLower(StringPiece16 string) { | 
 |   return CaseMap(string, &ToLowerMapper); | 
 | } | 
 |  | 
 | string16 ToUpper(StringPiece16 string) { | 
 |   return CaseMap(string, &ToUpperMapper); | 
 | } | 
 |  | 
 | string16 FoldCase(StringPiece16 string) { | 
 |   return CaseMap(string, &FoldCaseMapper); | 
 | } | 
 |  | 
 | }  // namespace i18n | 
 | }  // namespace base |