Android多语言适配(兼容7.0+)

一、前言

1、安卓系统本身对多语言适配就提供了一套框架和API。我们就直接用就可以了。
2、更换语言必须recreate Activity。目前,没见过可以不重建的方法。常用App,也都是重建的,可以看的到。
3、兼容性问题。现在越来越多设备都是安卓7.0+,新手机的安卓版本会更高(安卓8.0+),所以适配是必要的。
4、目前,网上大部分相关文章都是不兼容7.0+的,一搜一大把。

二、具体做法

1、多语言文件

文件夹命名参考下面博客(网上有很多):
多国语言value文件夹命名
value默认放英文的资源文件,简体中文文件夹命名为values-zh-rCN,不需要翻译的设置translatable如下:

<string name="app_name_english" translatable="false">You App English Name</string>

2、多语言工具类

public class LanguageUtils {
    public static final String CHINESE_SIMPLE = "zh_CN";
    public static final String ENGLISH = "en";
    public static final String AUTO = "auto";
    private static final String TAG = "LanguageUtils";
    //public static final String[] LOCALES = Utils.getContext().getResources().getStringArray(R.array.locales);

    private LanguageUtils() {
        throw new UnsupportedOperationException("u can't instantiate me...");
    }

    public static void setSystemDefaultLocale(Locale locale) {

    }

    public static boolean isSetValue(Context context) {
        Locale currentLocale = context.getResources().getConfiguration().locale;
        return currentLocale.equals(getSetLocale());
    }

    private static Locale getSetLocale() {
        String locale = SPUtils.getInstance(BaseConstants.SP.NAME_APP_SETTINGS).getString(BaseConstants.SP.KEY_LANGUAGE, LanguageUtils.AUTO);
        if (locale.equals(LanguageUtils.AUTO)) {
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                return Resources.getSystem().getConfiguration().getLocales().get(0);//解决了获取系统默认错误的问题
            } else {
                return Locale.getDefault();
            }
        }
        String[] array = locale.split("_");
        String language = array[0];
        if (array.length > 1) {
            String country = array[1];
            return new Locale(language, country);
        }
        return new Locale(language);
    }

    public static int getSetIndex() {
        String languageSet = SPUtils.getInstance(BaseConstants.SP.NAME_APP_SETTINGS).getString(BaseConstants.SP.KEY_LANGUAGE, LanguageUtils.AUTO);
        int localeIndex = 0;
        switch (languageSet) {
            case LanguageUtils.AUTO:
                localeIndex = 0;
                break;
            case LanguageUtils.CHINESE_SIMPLE:
                localeIndex = 1;
                break;
            case LanguageUtils.ENGLISH:
                localeIndex = 2;
                break;
        }
        return localeIndex;
    }

    public static Context wrapContext(Context context) {
        Resources resources = context.getResources();
        Locale locale = LanguageUtils.getSetLocale();

        Configuration configuration = resources.getConfiguration();
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
            configuration.setLocale(locale);
            LocaleList localeList = new LocaleList(locale);
            LocaleList.setDefault(localeList);
            configuration.setLocales(localeList);
        } else {
            configuration.setLocale(locale);
        }
        return context.createConfigurationContext(configuration);
    }

    public static void applyChange(Context context) {
        Resources res = context.getResources();
        DisplayMetrics dm = res.getDisplayMetrics();
        Configuration conf = res.getConfiguration();

        Locale locale = getSetLocale();
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
            conf.setLocale(locale);
            LocaleList localeList = new LocaleList(locale);
            LocaleList.setDefault(localeList);
            conf.setLocales(localeList);
        } else {
            conf.setLocale(locale);
        }
        res.updateConfiguration(conf, dm);
    }
}

3、代码分析&兼容7.0+

3.1、如何获取系统的语言设置(7.0+选择auto,可以正确切换)

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                return Resources.getSystem().getConfiguration().getLocales().get(0);//解决了获取系统默认错误的问题
            } else {
                return Locale.getDefault();
            }

看到这篇文章的你,可能已经看过网上很多其他相关的文章,应该知道,7.0+系统有个很奇怪的地方:
如果你在app内切换了语言(比如说是英文),且该语言和系统的设置(比如说是中文)不同,那么你再次切换语言并选择auto时,通过Locale.getDefault()获取会错误,或者你通过LocaleList.get(0)也是错误的,你之前选择的语言(英文)排序被提前了。有些文章的解决方案是在app打开时持久化系统设置,这样你切换app的语言就不会影响你获取系统的设置,但这样没必要,太麻烦(应该是不知道上面的方法)。
7.0+的系统设置也看的出差别,以前,设置系统语言直接选择就可以了,现在你要先添加,然后再排序,排在第一个的才是系统显示的语言!

3.2、写个BaseActivity作为所有Activity父类

新建一个BaseActivity用于继承,重写:

    @Override
    protected void attachBaseContext(Context newBase) {
        super.attachBaseContext(LanguageUtils.wrapContext(newBase));
    }

然后在切换语言后,你要recreate Activity。这个在哪调用就看具体需求了。你可以像微信那样,清空栈,然后直接重启到主界面,也可以在设置界面recreate,但栈内其他Activity,也要想办法通知recreate。

3.3、屏蔽系统设置改变

如果app的语言选项不是auto,那么系统语言设置修改时,app就不应该跟着系统变,而是按照自己设置的语言显示。写一个类继承于Application(注意要在manifest配置哦,不然无效的)

public class MyApp extends Application {
 private Configuration deltaConfig;

@Override
    public void onConfigurationChanged(Configuration newConfig) {
        LogUtils.d(TAG, "调用了onConfigurationChanged");
        int diff = newConfig.diff(deltaConfig);
        String languageSet = SPUtils.getInstance(AppConstants.SP.NAME_APP_SETTINGS).getString(AppConstants.SP.KEY_LANGUAGE, LanguageUtils.AUTO);
        if (languageSet.equals(LanguageUtils.AUTO)) {//看app语言设置是不是auto,是的话不管,直接super
            super.onConfigurationChanged(newConfig);
            deltaConfig = newConfig;
        } else if (diff != ActivityInfo.CONFIG_LOCALE) {//这个Configuration更改是不是语言,不是的话,也不管
            super.onConfigurationChanged(newConfig);
            deltaConfig = newConfig;
        }
         //这里使系统设置语言无效
         //相当于省略了
         //else{
         //    return;
         //}
    }

 @Override
    public void onCreate() {
        super.onCreate();
       //app打开时记录系统设置
        deltaConfig = getApplicationContext().getResources().getConfiguration();
        LanguageUtils.applyChange(getApplicationContext());
        }
    }
}

3.4、其他问题

Application的Context也要更新

LanguageUtils.applyChange(context);
LanguageUtils.applyChange(context.getApplicationContext());

但即使这样,还是有点问题,主要在于:
如果Activity的Title你是在manifest中定义的,如下label:

<activity
            android:name=".ui.activity.AboutActivity"
            android:launchMode="singleTop"
            android:label="@string/lable_activity_about"
            android:theme="@style/AppTheme.NoActionBar"/>

那么,即使你更新了ApplicationContext,有些Activity也有可能不生效,而且每次都还不一样,这个没法复现(很迷)。不知道是不是系统bug(测试系统是一加3 氢OS 8.0),或者是有其他更好的写法?
针对这个问题,只要在activity oncreate() 里setTitle()就好了。这样是不会有什么问题的。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,172评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,346评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,788评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,299评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,409评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,467评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,476评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,262评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,699评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,994评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,167评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,827评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,499评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,149评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,387评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,028评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,055评论 2 352

推荐阅读更多精彩内容