|
|
|
#include "manager.hpp"
|
|
|
|
|
|
|
|
#include <set>
|
|
|
|
#include <unicode/errorcode.h>
|
|
|
|
|
|
|
|
#include <components/debug/debuglog.hpp>
|
|
|
|
#include <components/vfs/manager.hpp>
|
|
|
|
|
|
|
|
namespace l10n
|
|
|
|
{
|
|
|
|
|
|
|
|
void Manager::setPreferredLocales(const std::vector<std::string>& langs, bool gmstHasPriority)
|
|
|
|
{
|
|
|
|
mPreferredLocales.clear();
|
|
|
|
if (gmstHasPriority)
|
|
|
|
mPreferredLocales.push_back(icu::Locale("gmst"));
|
|
|
|
std::set<std::string> langSet;
|
|
|
|
for (const auto& lang : langs)
|
|
|
|
{
|
|
|
|
if (langSet.contains(lang))
|
|
|
|
continue;
|
|
|
|
langSet.insert(lang);
|
|
|
|
mPreferredLocales.push_back(icu::Locale(lang.c_str()));
|
|
|
|
}
|
|
|
|
if (!gmstHasPriority)
|
|
|
|
mPreferredLocales.push_back(icu::Locale("gmst"));
|
|
|
|
{
|
|
|
|
Log msg(Debug::Info);
|
|
|
|
msg << "Preferred locales:";
|
|
|
|
for (const icu::Locale& l : mPreferredLocales)
|
|
|
|
msg << " " << l.getName();
|
|
|
|
}
|
|
|
|
for (auto& [key, context] : mCache)
|
|
|
|
updateContext(key.first, *context);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Manager::readLangData(const std::string& name, MessageBundles& ctx, const icu::Locale& lang)
|
|
|
|
{
|
|
|
|
std::string langName(lang.getName());
|
|
|
|
langName += ".yaml";
|
|
|
|
|
|
|
|
VFS::Path::Normalized path("l10n");
|
|
|
|
path /= name;
|
|
|
|
path /= langName;
|
|
|
|
|
|
|
|
const Files::IStreamPtr stream = mVFS->find(path);
|
|
|
|
|
|
|
|
if (stream == nullptr)
|
|
|
|
return;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
ctx.load(*stream, lang);
|
|
|
|
}
|
|
|
|
catch (const std::exception& e)
|
|
|
|
{
|
|
|
|
Log(Debug::Error) << "Cannot load message bundles from " << path << ": " << e.what();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Manager::updateContext(const std::string& name, MessageBundles& ctx)
|
|
|
|
{
|
|
|
|
icu::Locale fallbackLocale = ctx.getFallbackLocale();
|
|
|
|
ctx.setPreferredLocales(mPreferredLocales);
|
|
|
|
int localeCount = 0;
|
|
|
|
bool fallbackLocaleInPreferred = false;
|
|
|
|
for (const icu::Locale& loc : ctx.getPreferredLocales())
|
|
|
|
{
|
|
|
|
if (!ctx.isLoaded(loc))
|
|
|
|
readLangData(name, ctx, loc);
|
|
|
|
if (ctx.isLoaded(loc))
|
|
|
|
{
|
|
|
|
localeCount++;
|
|
|
|
Log(Debug::Verbose) << "Language file \"l10n/" << name << "/" << loc.getName() << ".yaml\" is enabled";
|
|
|
|
if (loc == ctx.getFallbackLocale())
|
|
|
|
fallbackLocaleInPreferred = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!ctx.isLoaded(ctx.getFallbackLocale()))
|
|
|
|
readLangData(name, ctx, ctx.getFallbackLocale());
|
|
|
|
if (ctx.isLoaded(ctx.getFallbackLocale()) && !fallbackLocaleInPreferred)
|
|
|
|
Log(Debug::Verbose) << "Fallback language file \"l10n/" << name << "/" << ctx.getFallbackLocale().getName()
|
|
|
|
<< ".yaml\" is enabled";
|
|
|
|
|
|
|
|
if (localeCount == 0)
|
|
|
|
{
|
|
|
|
Log(Debug::Warning) << "No language files for the preferred languages found in \"l10n/" << name << "\"";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::shared_ptr<const MessageBundles> Manager::getContext(
|
|
|
|
const std::string& contextName, const std::string& fallbackLocaleName)
|
|
|
|
{
|
|
|
|
std::pair<std::string, std::string> key(contextName, fallbackLocaleName);
|
|
|
|
auto it = mCache.find(key);
|
|
|
|
if (it != mCache.end())
|
|
|
|
return it->second;
|
|
|
|
auto allowedChar = [](char c) {
|
|
|
|
return (c >= 'A' && c <= 'Z') || (c >= 'a' && c <= 'z') || (c >= '0' && c <= '9') || c == '_';
|
|
|
|
};
|
|
|
|
bool valid = !contextName.empty();
|
|
|
|
for (char c : contextName)
|
|
|
|
valid = valid && allowedChar(c);
|
|
|
|
if (!valid)
|
|
|
|
throw std::runtime_error(std::string("Invalid l10n context name: ") + contextName);
|
|
|
|
icu::Locale fallbackLocale(fallbackLocaleName.c_str());
|
|
|
|
std::shared_ptr<MessageBundles> ctx = std::make_shared<MessageBundles>(mPreferredLocales, fallbackLocale);
|
|
|
|
ctx->setGmstLoader(mGmstLoader);
|
|
|
|
updateContext(contextName, *ctx);
|
|
|
|
mCache.emplace(key, ctx);
|
|
|
|
return ctx;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|