前言
最近接手一个Android项目,需要实现对维吾尔族语的支持。虽然做了这么久的android开发,只做过多语言支持,但做应用内部多语言支持还是第一次,而且还是对维吾尔语的支持。所以,又是一次面向搜索引擎编程。
面向搜索编程
如果我们搜索“android 多语言切换”,我相信得到大部分的答案是千篇一律的,连文章都长的一样:
Locale locale = new Locale("zh");
Locale.setDefault(locale);
Configuration config = new Configuration();
config.locale = locale;
getBaseContext().getResources().updateConfiguration(config, getBaseContext().getResources().getDisplayMetrics());
以上代码应该说没什么大问题,毕竟确实可以工作,可以实现需求。但是,作为一个强迫症患者,我实际受不了有2处划线的地方,也就是说上面代码中有2处被废弃了(没错,API 25被废弃的):
config.locale = locale;
getBaseContext().getResources().updateConfiguration(config,
getBaseContext().getResources().getDisplayMetrics());
看到有2处被废弃了,必须进去看看啊。
/**
* Current user preference for the locale, corresponding to
* <a href="{@docRoot}guide/topics/resources/providing-resources.html#LocaleQualifier">locale</a>
* resource qualifier.
*
* @deprecated Do not set or read this directly. Use {@link #getLocales()} and
* {@link #setLocales(LocaleList)}. If only the primary locale is needed,
* <code>getLocales().get(0)</code> is now the preferred accessor.
*/
@Deprecated public Locale locale;
很显然我们找到了替代方法:
config.setLocales(LocaleList);
config.setLocale(locale);
很好,没问题。继续查看下一个:
/**
* Store the newly updated configuration.
*
* @deprecated See {@link android.content.Context#createConfigurationContext(Configuration)}.
*/
@Deprecated
public void updateConfiguration(Configuration config, DisplayMetrics metrics) {
updateConfiguration(config, metrics, null);
}
很简单嘛,代替方法也找到了哦。于是就这么写了下去:
Locale locale = new Locale("ug");
final Resources res = getResources();
final Configuration config = res.getConfiguration();
config.setLocale(locale); // getLocale() should return a Locale
createConfigurationContext(config);
完全不好用!怎么写就是不好用。再仔细一看,文档上说createConfigurationContext(config)
会返回一个新的Context,然而对新Context如何处理只字未提。又是一番面向搜索编程,终于找到了正确姿势:
@Override
protected void attachBaseContext(Context newBase) {
Locale locale = new Locale("ug");
final Resources res = newBase.getResources();
final Configuration config = res.getConfiguration();
config.setLocale(locale); // getLocale() should return a Locale
final Context newContext = newBase.createConfigurationContext(config);
super.attachBaseContext(newContext);
}
我们的Activity应该重写这个方法。然后当Activity被创建的时候,新的Context将被应用。需要注意的,如果你想让当前Activity生效,你需要调用recreate()
。
当然啦,我们也可以写个工具类:
public class ConfigurationWrapper {
private ConfigurationWrapper() {
}
public static Context wrapConfiguration(@NonNull final Context context, @NonNull final Configuration config) {
return context.createConfigurationContext(config);
}
public static Context wrapLocale(@NonNull final Context context,@NonNull final Locale locale) {
final Resources res = context.getResources();
final Configuration config = res.getConfiguration();
config.setLocale(locale);
return wrapConfiguration(context, config);
}
}
然后你就可以这样使用。如果你想更改configuration,就可以用wrapConfiguration
。
@Override
protected void attachBaseContext(Context newBase) {
super.attachBaseContext(ConfigurationWrapper.wrapLocale(newContext, getLocale()));
}
关于Locale
实现多语言切换用到了Locale。Locale里很多常见国家和地区以及语言,如果我们做常见的语言,可以直接调用系统的,比如Locale.CHINESE
。但是这次做的维吾尔语的适配,略我坑一下。
首先,维吾尔语的英文名字叫Uyghur。Locale里并没有,也可能是我找的不对。
其次,Uyghur是阿拉伯系的。嗯,没错,有个词叫RTL,会出现神奇的东西。
最后,AndroidStudio的values里面是有Uyghur选项的。并且有相对应选项,其中就包括中国,我猜这就是维吾尔语。所以就建了一个,全名是:values-ug-rCN
。
问题是,我最开始直接这样写的:
Locale locale = new Locale("ug-rCN");
好吧,是我太天真,不好用。机智的我看了看Locale的构造方法。后来猜了一个结论
ug代表语种,rCN代表国家和地区。比如说同样都是中文zh,但分了很多国家和地区,如中国以及新家坡等
所以下面的写法才是正确姿势:
Locale locale = new Locale("ug", Locale.CHINA.getCountry());