Android: как получить строку в определенной локали БЕЗ изменения текущей локали
Вариант использования: регистрация сообщений об ошибках, отображаемых для пользователя.
Однако вы не хотите, чтобы в вашем журнале были сообщения, зависящие от локали устройства пользователя. С другой стороны, вы не хотите менять локаль устройства пользователя только для вашей (технической) регистрации. Можно ли этого достичь? Я нашел несколько потенциальных решений здесь на stackru:
- как получить строку из разных локалей в Android?
- Могу ли я получить доступ к ресурсам из разных локалей android?
- android - получает строку из локали по умолчанию, используя строку в определенной локали
- может быть, даже Android - Изменение локали в самом приложении
Однако все они приводят к изменению локали моего устройства (до следующего изменения конфигурации).
Во всяком случае, мой текущий обходной путь выглядит так:
public String getStringInDefaultLocale(int resId) {
Resources currentResources = getResources();
AssetManager assets = currentResources.getAssets();
DisplayMetrics metrics = currentResources.getDisplayMetrics();
Configuration config = new Configuration(
currentResources.getConfiguration());
config.locale = DEFAULT_LOCALE;
/*
* Note: This (temporiarily) changes the devices locale! TODO find a
* better way to get the string in the specific locale
*/
Resources defaultLocaleResources = new Resources(assets, metrics,
config);
String string = defaultLocaleResources.getString(resId);
// Restore device-specific locale
new Resources(assets, metrics, currentResources.getConfiguration());
return string;
}
Если честно, мне не нравится такой подход вообще. Это не эффективно и - думая о параллелизме и всем остальном - это может привести к некоторому взгляду на "неправильную" локаль.
Итак - есть идеи? Может быть, это может быть достигнуто с помощью ResourceBundle, как в стандартной Java?
4 ответа
Вы можете использовать это для API +17
@NonNull
@TargetApi(Build.VERSION_CODES.JELLY_BEAN_MR1)
public static String getStringByLocal(Activity context, int id, String locale) {
Configuration configuration = new Configuration(context.getResources().getConfiguration());
configuration.setLocale(new Locale(locale));
return context.createConfigurationContext(configuration).getResources().getString(id);
}
Обновление (1): Как поддерживать старые версии.
@NonNull
public static String getStringByLocal(Activity context, int resId, String locale) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1)
return getStringByLocalPlus17(context, resId, locale);
else
return getStringByLocalBefore17(context, resId, locale);
}
@NonNull
@TargetApi(Build.VERSION_CODES.JELLY_BEAN_MR1)
private static String getStringByLocalPlus17(Activity context, int resId, String locale) {
Configuration configuration = new Configuration(context.getResources().getConfiguration());
configuration.setLocale(new Locale(locale));
return context.createConfigurationContext(configuration).getResources().getString(resId);
}
private static String getStringByLocalBefore17(Context context,int resId, String language) {
Resources currentResources = context.getResources();
AssetManager assets = currentResources.getAssets();
DisplayMetrics metrics = currentResources.getDisplayMetrics();
Configuration config = new Configuration(currentResources.getConfiguration());
Locale locale = new Locale(language);
Locale.setDefault(locale);
config.locale = locale;
/*
* Note: This (temporarily) changes the devices locale! TODO find a
* better way to get the string in the specific locale
*/
Resources defaultLocaleResources = new Resources(assets, metrics, config);
String string = defaultLocaleResources.getString(resId);
// Restore device-specific locale
new Resources(assets, metrics, currentResources.getConfiguration());
return string;
}
Обновление (2): проверьте эту статью
Благодаря ответу @Khaled Lela я сделал расширение Kotlin:
fun Context.getStringByLocale(@StringRes stringRes: Int, locale: Locale, vararg formatArgs: Any): String {
val configuration = Configuration(resources.configuration)
configuration.setLocale(locale)
return createConfigurationContext(configuration).resources.getString(stringRes, *formatArgs)
}
И я использую прямо в Activity
, Fragment
или Context
связанные классы, просто вызывая:
getStringByLocale(R.string.my_text, Locale.UK)
или с аргументами:
getStringByLocale(R.string.my_other_text, Locale.RU, arg1, arg2, arg3)
Вы можете сохранить все строки локали по умолчанию в Map
внутри глобального класса, как Application
это выполняется при запуске приложения:
public class DualLocaleApplication extends Application {
private static Map<Integer, String> defaultLocaleString;
public void onCreate() {
super.onCreate();
Resources currentResources = getResources();
AssetManager assets = currentResources.getAssets();
DisplayMetrics metrics = currentResources.getDisplayMetrics();
Configuration config = new Configuration(
currentResources.getConfiguration());
config.locale = Locale.ENGLISH;
new Resources(assets, metrics, config);
defaultLocaleString = new HashMap<Integer, String>();
Class<?> stringResources = R.string.class;
for (Field field : stringResources.getFields()) {
String packageName = getPackageName();
int resId = getResources().getIdentifier(field.getName(), "string", packageName);
defaultLocaleString.put(resId, getString(resId));
}
// Restore device-specific locale
new Resources(assets, metrics, currentResources.getConfiguration());
}
public static String getStringInDefaultLocale(int resId) {
return defaultLocaleString.get(resId);
}
}
Это решение не является оптимальным, но у вас не будет проблем с параллелизмом.
Для API +17 мы могли бы использовать это:
public static String getDefaultString(Context context, @StringRes int stringId){
Resources resources = context.getResources();
Configuration configuration = new Configuration(resources.getConfiguration());
Locale defaultLocale = new Locale("en");
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
LocaleList localeList = new LocaleList(defaultLocale);
configuration.setLocales(localeList);
return context.createConfigurationContext(configuration).getString(stringId);
} else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1){
configuration.setLocale(defaultLocale);
return context.createConfigurationContext(configuration).getString(stringId);
}
return context.getString(stringId);
}
Попробуй это:
Configuration conf = getResources().getConfiguration();
conf.locale = new Locale("ar"); // locale I used here is Arabic
Resources resources = new Resources(getAssets(), getResources().getDisplayMetrics(), conf);
/* get localized string */
String appName = resources.getString(R.string.app_name);