接到一个需求,在app设置里面添加一个功能,改变app语言,app做了多国语言适配,一开始我的想法是在application中写若干代码,改变整个项目的语言环境,我搜到的方法大致是这样的
1. 添加多语言文件
在不同的 value 文件夹下(例如 value 、value-en、values-zh-rTW 文件夹)添加不同语言的 string.xml
文件,我们的项目添加了英文、简体中文、繁体中文三种语言,如下图所示:
其中英文需要翻译,繁体如果没有专门翻译的话,可以找个简繁转换网站,直接将简体中文转成繁体中文,我用的这个网站: 在线中文简体转繁体 。
2. 更新 Configuration 中的 locale 属性
参照 Android 开发者官网 上的描述,Configuration 包含了设备的所有的配置信息,这些配置信息会影响应用获取的资源。例如 string 资源,就是根据 Configuration 的 locale 属性来判断该取哪种语言的 string 资源,默认是 value 文件夹下的。
主要代码如下:
Resources resources = getContext().getResources();
DisplayMetrics dm = resources.getDisplayMetrics();
Configuration config = resources.getConfiguration();
// 应用用户选择语言
config.locale = Locale.ENGLISH;
resources.updateConfiguration(config, dm);
我们用了 Locale
中的预设值 Locale.ENGLISH
、 Locale.TRADITIONAL_CHINESE
和 Locale.SIMPLIFIED_CHINESE
,如果你需要设置的语言没有预设值,你可以自己新建一个 Locale
对象,具体自行 Google 吧。
注:跟随系统设置是 Locale.getDefault()
3. 重启 HomeActivity
我们的 App 有个启动页 WelcomeActivity,类似微信那个小人启动页,如果从欢迎页重启,并不是一个好的体验,应该和微信的语言设置一样,直接回到 HomeActivity ,而不是从 WelcomeActivity 重新打开。实现其实也很简单,代码如下:
Intent intent = new Intent(this, HomeActivity.class);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TASK);
getActivity().startActivity(intent);
正常来说这段代码应该是没问题的,但是假如你的 App 存在某个 activity 和当前设置页 activity 不在一个 task 栈内的话(比如你从某个通知页用 FLAG_ACTIVITY_NEW_TASK
启动的一个 activity),就不会应用语言设置。因此可以直接杀掉当前 App 的进程,保证是“整个”重启了:
Intent intent = new Intent(this, HomeActivity.class);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TASK);
startActivity(intent);
// 杀掉进程
android.os.Process.killProcess(android.os.Process.myPid());
System.exit(0);
按道理杀掉进程的两行代码任意一行即可,但是查阅相关资料,还是两个都加上吧,如果有详细了解欢迎沟通。此段代码其实参考自 CustomActivityOnCrash 开源项目,有兴趣的可以研究下这个开源库捕捉崩溃信息,重启应用部分的代码。
持久化存储语言设置
按照上述三个步骤,你应该已经可以看到了改变语言的效果了,但是当你杀掉应用,重新打开,发现设置又失效了。这是因为应用重启后会读取设备默认的 Configuration 信息,其中和语言相关的 locale 属性也会变成默认值,也就是你在系统设置中选择的语言。
当你的应用需要由用户单独设置语言,而不是仅仅跟随系统语言,你就需要持久化存储用户的设置信息了。你可以选择数据库、或 SharedPreferences 来存储设置信息。
在应用启动时需要读取存储的设置,并应用该配置,简要代码如下:
public class App extends MultiDexApplication {
@Override
public void onCreate() {
super.onCreate();
...
Resources resources = App.getContext().getResources();
DisplayMetrics dm = resources.getDisplayMetrics();
Configuration config = resources.getConfiguration();
config.locale = getSetLocale();
resources.updateConfiguration(config, dm);
}
// 得到设置的语言信息
private static Locale getSetLocale() {
// 读取储存的语言设置信息
...
}
}
===============================================================
之前也试过网上的一些方法,但是效果总是差强人意,比如更改完app的语言之后只有当前跳转到的activity语言是更换的,下一个activity的语言又是跟随系统走的,在application中设置的也是如此,这点我一直很疑惑,希望有高人来帮忙解答。
我的实现方法是利用sharepreference
首先在更改语言的方法中写
private void changeLanguage(int index) { SharedPreferences preferences=getSharedPreferences("language",MODE_PRIVATE); preferences.edit().putInt("language",index).commit(); Intent intent=new Intent(this,DeviceListActivity.class); startActivity(intent); android.os.Process.killProcess(android.os.Process.myPid()); System.exit(0); }
index是一个数组,里面存放了我的可适配的语言,当然也可以放locale
然后在基类baseactivity的oncreate方法中写上
@Override protected void onCreate(@Nullable Bundle savedInstanceState) { super.onCreate(savedInstanceState); Configuration config = getResources().getConfiguration(); DisplayMetrics metrics = getResources().getDisplayMetrics(); SharedPreferences preferences=getSharedPreferences("language",MODE_PRIVATE); int language = preferences.getInt("language", 8); Log.d("tag",language+""); switch (language){ case 0: config.locale = Locale.ENGLISH; break; case 1: config.locale= Locale.SIMPLIFIED_CHINESE; break; case 2: config.locale= Locale.TRADITIONAL_CHINESE; break; case 3: config.locale= Locale.ENGLISH; break; case 4: config.locale=Locale.GERMANY; break; case 5: config.locale=Locale.ENGLISH; break; case 6: config.locale= Locale.FRANCE; break; case 7: config.locale= Locale.JAPAN; break; case 8: break; } getResources().updateConfiguration(config, metrics); }
就可以很轻松的实现更换语言功能,但是这个方法也有个弊端,就是没有继承基类的activity就不受控制,例如一些引用第三方库的activity
===============================================================
接下来是介绍一个安卓修改app语言简体繁体一个比较详细的例子:
http://blog.csdn.net/qq_30543115/article/details/52935726?locationNum=2&fps=1