#include "messagebundles.hpp" #include #include #include #include #include namespace l10n { MessageBundles::MessageBundles(const std::vector &preferredLocales, icu::Locale &fallbackLocale) : mFallbackLocale(fallbackLocale) { setPreferredLocales(preferredLocales); } void MessageBundles::setPreferredLocales(const std::vector &preferredLocales) { mPreferredLocales.clear(); mPreferredLocaleStrings.clear(); for (const icu::Locale &loc: preferredLocales) { mPreferredLocales.push_back(loc); mPreferredLocaleStrings.push_back(loc.getName()); // Try without variant or country if they are specified, starting with the most specific if (strcmp(loc.getVariant(), "") != 0) { icu::Locale withoutVariant(loc.getLanguage(), loc.getCountry()); mPreferredLocales.push_back(withoutVariant); mPreferredLocaleStrings.push_back(withoutVariant.getName()); } if (strcmp(loc.getCountry(), "") != 0) { icu::Locale withoutCountry(loc.getLanguage()); mPreferredLocales.push_back(withoutCountry); mPreferredLocaleStrings.push_back(withoutCountry.getName()); } } } std::string getErrorText(const UParseError &parseError) { icu::UnicodeString preContext(parseError.preContext), postContext(parseError.postContext); std::string parseErrorString; preContext.toUTF8String(parseErrorString); postContext.toUTF8String(parseErrorString); return parseErrorString; } static bool checkSuccess(const icu::ErrorCode &status, const std::string &message, const UParseError parseError = UParseError()) { if (status.isFailure()) { std::string errorText = getErrorText(parseError); if (errorText.size()) { Log(Debug::Error) << message << ": " << status.errorName() << " in \"" << errorText << "\""; } else { Log(Debug::Error) << message << ": " << status.errorName(); } } return status.isSuccess(); } void MessageBundles::load(std::istream &input, const icu::Locale& lang, const std::string &path) { try { YAML::Node data = YAML::Load(input); std::string localeName = lang.getName(); for (const auto& it: data) { std::string key = it.first.as(); std::string value = it.second.as(); icu::UnicodeString pattern = icu::UnicodeString::fromUTF8(value); icu::ErrorCode status; UParseError parseError; icu::MessageFormat message(pattern, lang, parseError, status); if (checkSuccess(status, std::string("Failed to create message ") + key + " for locale " + lang.getName(), parseError)) { mBundles[localeName].insert(std::make_pair(key, message)); } } } catch (std::exception& e) { Log(Debug::Error) << "Can not load " << path << ": " << e.what(); } } const icu::MessageFormat * MessageBundles::findMessage(std::string_view key, const std::string &localeName) const { auto iter = mBundles.find(localeName); if (iter != mBundles.end()) { auto message = iter->second.find(key.data()); if (message != iter->second.end()) { return &(message->second); } } return nullptr; } std::string MessageBundles::formatMessage(std::string_view key, const std::map &args) const { std::vector argNames; std::vector argValues; for (auto& [key, value] : args) { argNames.push_back(icu::UnicodeString::fromUTF8(key)); argValues.push_back(value); } return formatMessage(key, argNames, argValues); } std::string MessageBundles::formatMessage(std::string_view key, const std::vector &argNames, const std::vector &args) const { icu::UnicodeString result; std::string resultString; icu::ErrorCode success; const icu::MessageFormat *message = nullptr; for (auto &loc: mPreferredLocaleStrings) { message = findMessage(key, loc); if (message) break; } // If no requested locales included the message, try the fallback locale if (!message) message = findMessage(key, mFallbackLocale.getName()); if (message) { if (args.size() > 0 && argNames.size() > 0) message->format(&argNames[0], &args[0], args.size(), result, success); else message->format(nullptr, nullptr, args.size(), result, success); checkSuccess(success, std::string("Failed to format message ") + key.data()); result.toUTF8String(resultString); return resultString; } icu::Locale defaultLocale(NULL); if (mPreferredLocales.size() > 0) { defaultLocale = mPreferredLocales[0]; } UParseError parseError; icu::MessageFormat defaultMessage(icu::UnicodeString::fromUTF8(key), defaultLocale, parseError, success); if (!checkSuccess(success, std::string("Failed to create message ") + key.data(), parseError)) // If we can't parse the key as a pattern, just return the key return std::string(key); if (args.size() > 0 && argNames.size() > 0) defaultMessage.format(&argNames[0], &args[0], args.size(), result, success); else defaultMessage.format(nullptr, nullptr, args.size(), result, success); checkSuccess(success, std::string("Failed to format message ") + key.data()); result.toUTF8String(resultString); return resultString; } }