|  | // Copyright 2015 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/message_formatter.h" | 
|  |  | 
|  | #include "base/i18n/unicodestring.h" | 
|  | #include "base/logging.h" | 
|  | #include "base/numerics/safe_conversions.h" | 
|  | #include "base/time/time.h" | 
|  | #include "third_party/icu/source/common/unicode/unistr.h" | 
|  | #include "third_party/icu/source/common/unicode/utypes.h" | 
|  | #include "third_party/icu/source/i18n/unicode/fmtable.h" | 
|  | #include "third_party/icu/source/i18n/unicode/msgfmt.h" | 
|  |  | 
|  | using icu::UnicodeString; | 
|  |  | 
|  | namespace base { | 
|  | namespace i18n { | 
|  | namespace { | 
|  | UnicodeString UnicodeStringFromStringPiece(StringPiece str) { | 
|  | return UnicodeString::fromUTF8( | 
|  | icu::StringPiece(str.data(), base::checked_cast<int32_t>(str.size()))); | 
|  | } | 
|  | }  // anonymous namespace | 
|  |  | 
|  | namespace internal { | 
|  | MessageArg::MessageArg() : formattable(nullptr) {} | 
|  |  | 
|  | MessageArg::MessageArg(const char* s) | 
|  | : formattable(new icu::Formattable(UnicodeStringFromStringPiece(s))) {} | 
|  |  | 
|  | MessageArg::MessageArg(StringPiece s) | 
|  | : formattable(new icu::Formattable(UnicodeStringFromStringPiece(s))) {} | 
|  |  | 
|  | MessageArg::MessageArg(const std::string& s) | 
|  | : formattable(new icu::Formattable(UnicodeString::fromUTF8(s))) {} | 
|  |  | 
|  | MessageArg::MessageArg(const string16& s) | 
|  | : formattable(new icu::Formattable(UnicodeString(s.data(), s.size()))) {} | 
|  |  | 
|  | MessageArg::MessageArg(int i) : formattable(new icu::Formattable(i)) {} | 
|  |  | 
|  | MessageArg::MessageArg(int64_t i) : formattable(new icu::Formattable(i)) {} | 
|  |  | 
|  | MessageArg::MessageArg(double d) : formattable(new icu::Formattable(d)) {} | 
|  |  | 
|  | MessageArg::MessageArg(const Time& t) | 
|  | : formattable(new icu::Formattable(static_cast<UDate>(t.ToJsTime()))) {} | 
|  |  | 
|  | MessageArg::~MessageArg() = default; | 
|  |  | 
|  | // Tests if this argument has a value, and if so increments *count. | 
|  | bool MessageArg::has_value(int *count) const { | 
|  | if (formattable == nullptr) | 
|  | return false; | 
|  |  | 
|  | ++*count; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | }  // namespace internal | 
|  |  | 
|  | string16 MessageFormatter::FormatWithNumberedArgs( | 
|  | StringPiece16 msg, | 
|  | const internal::MessageArg& arg0, | 
|  | const internal::MessageArg& arg1, | 
|  | const internal::MessageArg& arg2, | 
|  | const internal::MessageArg& arg3, | 
|  | const internal::MessageArg& arg4, | 
|  | const internal::MessageArg& arg5, | 
|  | const internal::MessageArg& arg6) { | 
|  | int32_t args_count = 0; | 
|  | icu::Formattable args[] = { | 
|  | arg0.has_value(&args_count) ? *arg0.formattable : icu::Formattable(), | 
|  | arg1.has_value(&args_count) ? *arg1.formattable : icu::Formattable(), | 
|  | arg2.has_value(&args_count) ? *arg2.formattable : icu::Formattable(), | 
|  | arg3.has_value(&args_count) ? *arg3.formattable : icu::Formattable(), | 
|  | arg4.has_value(&args_count) ? *arg4.formattable : icu::Formattable(), | 
|  | arg5.has_value(&args_count) ? *arg5.formattable : icu::Formattable(), | 
|  | arg6.has_value(&args_count) ? *arg6.formattable : icu::Formattable(), | 
|  | }; | 
|  |  | 
|  | UnicodeString msg_string(msg.data(), msg.size()); | 
|  | UErrorCode error = U_ZERO_ERROR; | 
|  | icu::MessageFormat format(msg_string,  error); | 
|  | icu::UnicodeString formatted; | 
|  | icu::FieldPosition ignore(icu::FieldPosition::DONT_CARE); | 
|  | format.format(args, args_count, formatted, ignore, error); | 
|  | if (U_FAILURE(error)) { | 
|  | LOG(ERROR) << "MessageFormat(" << msg.as_string() << ") failed with " | 
|  | << u_errorName(error); | 
|  | return string16(); | 
|  | } | 
|  | return i18n::UnicodeStringToString16(formatted); | 
|  | } | 
|  |  | 
|  | string16 MessageFormatter::FormatWithNamedArgs( | 
|  | StringPiece16 msg, | 
|  | StringPiece name0, const internal::MessageArg& arg0, | 
|  | StringPiece name1, const internal::MessageArg& arg1, | 
|  | StringPiece name2, const internal::MessageArg& arg2, | 
|  | StringPiece name3, const internal::MessageArg& arg3, | 
|  | StringPiece name4, const internal::MessageArg& arg4, | 
|  | StringPiece name5, const internal::MessageArg& arg5, | 
|  | StringPiece name6, const internal::MessageArg& arg6) { | 
|  | icu::UnicodeString names[] = { | 
|  | UnicodeStringFromStringPiece(name0), | 
|  | UnicodeStringFromStringPiece(name1), | 
|  | UnicodeStringFromStringPiece(name2), | 
|  | UnicodeStringFromStringPiece(name3), | 
|  | UnicodeStringFromStringPiece(name4), | 
|  | UnicodeStringFromStringPiece(name5), | 
|  | UnicodeStringFromStringPiece(name6), | 
|  | }; | 
|  | int32_t args_count = 0; | 
|  | icu::Formattable args[] = { | 
|  | arg0.has_value(&args_count) ? *arg0.formattable : icu::Formattable(), | 
|  | arg1.has_value(&args_count) ? *arg1.formattable : icu::Formattable(), | 
|  | arg2.has_value(&args_count) ? *arg2.formattable : icu::Formattable(), | 
|  | arg3.has_value(&args_count) ? *arg3.formattable : icu::Formattable(), | 
|  | arg4.has_value(&args_count) ? *arg4.formattable : icu::Formattable(), | 
|  | arg5.has_value(&args_count) ? *arg5.formattable : icu::Formattable(), | 
|  | arg6.has_value(&args_count) ? *arg6.formattable : icu::Formattable(), | 
|  | }; | 
|  |  | 
|  | UnicodeString msg_string(msg.data(), msg.size()); | 
|  | UErrorCode error = U_ZERO_ERROR; | 
|  | icu::MessageFormat format(msg_string, error); | 
|  |  | 
|  | icu::UnicodeString formatted; | 
|  | format.format(names, args, args_count, formatted, error); | 
|  | if (U_FAILURE(error)) { | 
|  | LOG(ERROR) << "MessageFormat(" << msg.as_string() << ") failed with " | 
|  | << u_errorName(error); | 
|  | return string16(); | 
|  | } | 
|  | return i18n::UnicodeStringToString16(formatted); | 
|  | } | 
|  |  | 
|  | }  // namespace i18n | 
|  | }  // namespace base |