Android 内多语言切换实现

举报
xq9527 发表于 2022/06/09 20:47:08 2022/06/09
【摘要】 前言:网上有很多安卓内多语言切换的文章,我也看了一些,很多都千篇一律,而且还是有问题,我就自己重新改了一下。整好分享给各位同学,废话不多说,我们正式开始 效果图我们在点击切换语言的按钮时候弹窗dialog 然后用户可以选择 简体中文 英文 繁体中文 跟随系统 四种选项 ,这是我们基本的需求 具体实现首先我们要在资源文件res目录下面创建多语言的目录文件夹 values valu...

前言:

网上有很多安卓内多语言切换的文章,我也看了一些,很多都千篇一律,而且还是有问题,我就自己重新改了一下。整好分享给各位同学,废话不多说,我们正式开始

效果图

6865547-353a3de56538bdb5.png
6865547-1ad75e454dbd42f6.png
6865547-ff68e829eb148256.png
我们在点击切换语言的按钮时候弹窗dialog 然后用户可以选择 简体中文 英文 繁体中文 跟随系统 四种选项 ,这是我们基本的需求

具体实现

首先我们要在资源文件res目录下面创建多语言的目录文件夹 values values-en values-zh-rTW values-TW 如图所示:
6865547-20477bbc0ad395d1.png
我们适配多语言的只需要在 values values-en values-zh-rTW 这个三个目录下面的string.xml文件里面写入对应的引用的文本即可

简体中文资源文件配置

<resources>
    <string name="app_name">Language Demo</string>
    <string name="lan_chinese">中文</string>
    <string name="lan_en">英文</string>
    <string name="text_content">这是内容显示区</string>
    <string name="select_language">选择一个语言</string>
    <string name="lan_zh_rTYW">繁体中文</string>
    <string name="Follow_the_system">跟随系統</string>
</resources>

英文资源文件配置

<resources>
    <string name="app_name">Language Demo</string>
    <string name="lan_chinese">Chinese</string>
    <string name="lan_en">English</string>
    <string name="text_content">Hello World!</string>
    <string name="select_language">Choose a language</string>
    <string name="lan_zh_rTYW">Chinese Traditional</string>
    <string name="Follow_the_system">Follow the system</string>
</resources>

繁体中文资源文件配置

<resources>
<resources>
    <string name="app_name">Language Demo</string>
    <string name="lan_chinese">簡體中文</string>
    <string name="lan_en">英文</string>
    <string name="lan_zh_rTYW">繁體中文</string>
    <string name="text_content">這是內容顯示區域</string>
    <string name="select_language">選擇一種語言</string>
    <string name="Follow_the_system">跟隨系統</string>
</resources>
</resources>

我们只需要这样配置就然后在布局的xml文件里面引用可以达到适配多语言的效果 :
具体引用:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context=".MainActivity">
<TextView
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="@string/text_content"
    android:id="@+id/textView" />
<Button
    android:id="@+id/btn_setting"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="setting" />
</LinearLayout>

切换的dialog实现这个我用原生的AlertDialog 你们自己可以自定义继承系统的Dialog来实现

        final String[] cities = {getString(R.string.lan_chinese), getString(R.string.lan_en),getString(R.string.lan_zh_rTYW),getString(R.string.Follow_the_system)};
        final String[] locals = {"zh_CN", "en","zh_TW","111"};
        Button button = (Button)findViewById(R.id.btn_setting);
        button.setText("Language");
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                AlertDialog.Builder builder = new AlertDialog.Builder(SettingActivity.this);
                builder.setIcon(R.mipmap.ic_launcher);
                builder.setTitle(R.string.select_language);
                builder.setItems(cities, new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialog, int which) {
                        String  language=null;
                        if(which==3){
                            language= LanguageUtils.getCurrentLanguage();
                            Log.e(TAG, "onClick: language   --- >"+language  );
                        }else{
                            language=locals[which];
                        }
                        Store.setLanguageLocal(SettingActivity.this, language);
                        Intent intent = new Intent(Config.ACTION);
                        intent.putExtra("msg", "EVENT_REFRESH_LANGUAGE");
                        sendBroadcast(intent);

                    }
                });
                builder.show();

            }
        });
更新 Configuration 中的 locale 属性
    public void changeAppLanguage() {
        String sta = Store.getLanguageLocal(this);
        if(sta != null && !"".equals(sta)){
            // 本地语言设置
            Locale myLocale=null;
            if(sta.equals("zh_CN")){
                myLocale = new Locale(sta,Locale.CHINESE.getCountry());
            }else if(sta.equals("zh_TW")){
                myLocale = new Locale("TW",Locale.TRADITIONAL_CHINESE.getCountry());
            }else  if(sta.equals("en")||sta.equals("en_US")){
                myLocale = new Locale( "en",Locale.ENGLISH.getCountry());
            }
            Resources res = getResources();
            DisplayMetrics dm = res.getDisplayMetrics();
            Configuration conf = res.getConfiguration();
            conf.locale = myLocale;
            res.updateConfiguration(conf, dm);
        }
    }

这里的语言种类获取到的sta是重SharedPreferences 缓存里面获取到的因为需要
SharedPreferences 工具类:

package com.language_demo;
import android.content.Context;
import android.content.SharedPreferences;
import android.preference.PreferenceManager;
public class Store {
    public static void setLanguageLocal(Context context, String language){
        SharedPreferences preferences;
        SharedPreferences.Editor editor;
        preferences = PreferenceManager.getDefaultSharedPreferences(context);
        editor = preferences.edit();
        editor.putString("language", language);
        editor.commit();
    }
    public static String getLanguageLocal(Context context){
        SharedPreferences preferences;
        preferences = PreferenceManager.getDefaultSharedPreferences(context);
        String language = preferences.getString("language", "");
        return language;
    }
}

我们在dialog点击事件里发送一个广播通知 用来触发刷新

    Store.setLanguageLocal(SettingActivity.this, language);
     Intent intent = new Intent(Config.ACTION);
     intent.putExtra("msg", "EVENT_REFRESH_LANGUAGE");
   sendBroadcast(intent);

然我们在baseActivity里面接收广播重启activity 来刷新多语言切换

    BroadcastReceiver myBroadcastReceive = new BroadcastReceiver() {
        @Override
        public void onReceive(Context context, Intent intent) {
            Log.i("广播", "----接收到的是----" + intent.getStringExtra("msg"));
            if(intent.getStringExtra("msg").equals("EVENT_REFRESH_LANGUAGE")){
                changeAppLanguage();
                recreate();//刷新界面
            }
        }
    };

关于Locale
实现多语言切换用到了Locale。Locale里很多常见国家和地区以及语言,如果我们做常见的语言,可以直接调用系统的,比如Locale.CHINESE。但是这次做的繁体中文的适配,略我坑一下。
繁体中文我们直接在Locale 属性里面传入 zh-rTW 但是切换没有生效 我翻了下源码只有TW的属性配置我就要修改了
问题是,我最开始直接这样写的:

Locale myLocale = new Locale(sta);

到此我们说明下开头提到到为什么多出了 values-TW目录 是因为我发现在多语言切换更新 Configuration 中的 locale 属性 的时候翻开源码看到繁体中文(泛指港澳台并没有看到zh-rTW的属性配置)

如图:

6865547-dc705d41a98b3a7c.png
所以目前的解决方案就是用values-TW来替代:

 myLocale = new Locale("TW",Locale.TRADITIONAL_CHINESE.getCountry());

我这边目前是这样处理来兼容繁体中文切换不生效的做法 ,如果你有更好的方案 麻烦留言大家一起探讨。
##最后总结:
Android 内多语言的切换网上的文章也挺多的,基本都千篇一律 很多有纰漏 ,这边也是翻了一下源码勉强能够兼容繁体中文,有兴趣学的的同学可以私下多多交流 最后希望我的文章能帮助到各位解决问题 ,以后我还会贡献更多有用的代码分享给大家。各位同学如果觉得文章还不错 ,麻烦给关注和star,小弟在这里谢过啦 也可以加我个人QQ/微信(1693891473)

项目地址:

码云 :https://gitee.com/qiuyu123/language_demo.git

QQ交流群:

6865547-4d8ef2c6980ca44e.png

【版权声明】本文为华为云社区用户原创内容,转载时必须标注文章的来源(华为云社区)、文章链接、文章作者等基本信息, 否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。