如何更改Android O / Oreo / api 26应用程序语言

时间:2017-11-07 18:45:37

标签: java android locale default

我想更改应用程序的语言,直到API 26才能正常工作。

对于api> 25我将Locale.setDefault(Locale.Category.DISPLAY, mynewlanglocale);放在setContentView(R.layout.activity_main);之前,但没有任何变化。

docs不要过多地解释这一点。

8 个答案:

答案 0 :(得分:17)

是的,在android中,Oreo本地化在updateconfiguration中运行不正常。但它在Android N本身被弃用了。而不是更新配置使用每个attachcontext中的createconfiguration。它对我来说很好。试试这个......

在你的活动中添加这个..

@Override
protected void attachBaseContext(Context newBase) {
    if(Build.VERSION.SDK_INT > Build.VERSION_CODES.N_MR1) {
        super.attachBaseContext(MyContextWrapper.wrap(newBase, "ta"));
    }
    else {
        super.attachBaseContext(newBase);
    }
}

在MyContextWrapper.java中

 public static ContextWrapper wrap(Context context, String language) {
    Resources res = context.getResources();
    Configuration configuration = res.getConfiguration();
    Locale newLocale = new Locale(language);

    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
        configuration.setLocale(newLocale);
        LocaleList localeList = new LocaleList(newLocale);
        LocaleList.setDefault(localeList);
        configuration.setLocales(localeList);
        context = context.createConfigurationContext(configuration);

    } else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1) {
        configuration.setLocale(newLocale);
        context = context.createConfigurationContext(configuration);

    } else {
        configuration.locale = newLocale;
        res.updateConfiguration(configuration, res.getDisplayMetrics());
    }

    return new ContextWrapper(context);
}

答案 1 :(得分:13)

我遇到了同样的问题:由于Android 8.0+,我的应用程序的某些部分不再更改其语言。更新应用程序和活动上下文对我有帮助。这是MainActivity函数的示例:

private void setApplicationLanguage(String newLanguage) {
    Resources activityRes = getResources();
    Configuration activityConf = activityRes.getConfiguration();
    Locale newLocale = new Locale(newLanguage);
    activityConf.setLocale(newLocale);
    activityRes.updateConfiguration(activityConf, activityRes.getDisplayMetrics());

    Resources applicationRes = getApplicationContext().getResources();
    Configuration applicationConf = applicationRes.getConfiguration();
    applicationConf.setLocale(newLocale);
    applicationRes.updateConfiguration(applicationConf, 
    applicationRes.getDisplayMetrics());
}

答案 2 :(得分:7)

不推荐使用

updateConfiguration,您应该使用createConfigurationContext。我这样解决了:

    @Override
    protected void attachBaseContext(Context newBase) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
            Configuration config = newBase.getResources().getConfiguration();
            //Update your config with the Locale i. e. saved in SharedPreferences
            SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(newBase);
            String language = prefs.getString(SP_KEY_LANGUAGE, "en_US");
            Locale.setDefault(locale);
            config.setLocale(new Locale(language));
            newBase = newBase.createConfigurationContext(config);
        }
        super.attachBaseContext(newBase);
    }

答案 3 :(得分:3)

更新了所有Android版本,直到奥利奥

创建一个这样的类

public class LocaleUtils {

@Retention(RetentionPolicy.SOURCE)
@StringDef({ENGLISH, FRENCH, SPANISH})
public @interface LocaleDef {
    String[] SUPPORTED_LOCALES = {ENGLISH, FRENCH, SPANISH};
}

public static final String ENGLISH = "en";
public static final String FRENCH = "fr";
public static final String SPANISH = "es";


public static void initialize(Context context) {
    setLocale(context, ENGLISH);
}

public static void initialize(Context context, @LocaleDef String defaultLanguage) {
    setLocale(context, defaultLanguage);
}


public static boolean setLocale(Context context, @LocaleDef String language) {
    return updateResources(context, language);
}

private static boolean updateResources(Context context, String language) {
    Locale locale = new Locale(language);
    Locale.setDefault(locale);
    Resources resources = context.getResources();
    Configuration configuration = resources.getConfiguration();
    context.createConfigurationContext(configuration);
    configuration.locale = locale;
    resources.updateConfiguration(configuration, resources.getDisplayMetrics());
    return true;
}

}

现在,当您从应用中选择语言时,请将语言代码保存在共享首选项中,如下所示

private static SharedPreferences getDefaultSharedPreference(Context context) {
    if (PreferenceManager.getDefaultSharedPreferences(Application.getInstance().getApplicationContext()) != null)
        return PreferenceManager.getDefaultSharedPreferences(Application.getInstance().getApplicationContext());
    else
        return null;
}

 public static void setSelectedLanguageId(String id){
    final SharedPreferences prefs = getDefaultSharedPreference(Application.getInstance().getApplicationContext());
    SharedPreferences.Editor editor = prefs.edit();
    editor.putString("app_language_id", id);
    editor.apply();
}

public static String getSelectedLanguageId(){
    return getDefaultSharedPreference(Application.getInstance().getApplicationContext())
            .getString("app_language_id", "en");
}

这三个函数应该写在Utiltiy类中(您的首选项)。然后,当您从应用程序中选择应用程序语言时,请调用setSelectedLanguageId()函数并将语言ID作为参数传递。

这样您就可以在应用中保存所选语言。现在在你的应用程序类中编写一个像这样的函数

public void initAppLanguage(Context context){
    LocaleUtils.initialize(context, PreferenceUtil.getSelectedLanguageId() );
}

这里的PreferenceUtil是我的Utiltiy类。您应该用实用程序类函数替换它。

您还应该在应用程序类中创建一个变量

private static Application applicationInstance;

并在您的Application类的onCreate方法中,初始化applicationInstance为这样的应用程序上下文

applicationInstance = this; 

现在在应用程序类中编写一个getter函数

public static synchronized Application getInstance() {
    return applicationInstance;
}

现在,当您开始第一项活动时,请在您的活动“onCreate”中调用此方法

Application.getInstance().initAppLanguage(this);

请记住,我们将活动的上下文传递给initAppLanguage()函数,而不是应用程序上下文。传递应用程序上下文不会让它在Oreo中工作(至少对我而言)。

因此,当您选择语言时,请尝试完全重新启动应用程序。 你可以通过

实现这个目标
Intent i = getBaseContext().getPackageManager().getLaunchIntentForPackage(getBaseContext().getPackageName());

startActivity(ⅰ);

希望这能帮到你!

答案 4 :(得分:1)

有可能,但我不建议以编程方式设置语言

Android的设计使系统用户界面和您的应用具有相同的语言,如果您以编程方式更改它,那么您将与系统进行战斗

相反,您可以通过添加不同的strings.xml语言来启用多语言支持,这将自动更改语言

我建议您阅读 Google Developers 文章:

Supporting Different Languages and Cultures

如果您确实需要以编程方式更改它,可以执行以下操作

Locale locale = new Locale("en");
Locale.setDefault(locale);

Configuration config = context.getResources().getConfiguration();
config.setLocale(locale);
context.createConfigurationContext(config);
context.getResources().updateConfiguration(config, context.getResources().getDisplayMetrics());

在SDK> = 21上,您需要调用'Resources.updateConfiguration()',否则资源将不会更新。

希望它有所帮助。

答案 5 :(得分:1)

以下是kitkat,Lollipop,Marshmallow,Nougat和Oreo的完整解决方案。只需按照以下步骤操作即可。

首先创建一个如下所示的java类

import android.content.Context;
import android.content.res.Configuration;
import java.util.Locale;
public class LocaleUtils {
public static void updateConfig(Context mContext, String sLocale) {
    Locale locale = new Locale(sLocale);
    Locale.setDefault(locale);
    Configuration config = mContext.getResources().getConfiguration();
    config.locale = locale;
    mContext.getResources().updateConfiguration(config,
            mContext.getResources().getDisplayMetrics());
  }
}

现在,在按钮上添加此片段,然后单击要更改区域设置的位置

String lang="hi";//pass your language here
SharedPreferences preferences =  PreferenceManager.getDefaultSharedPreferences(mContext);
                        SharedPreferences.Editor editor = preferences.edit();
                        editor.clear();
                        editor.putString("lang", lang");
                        editor.putBoolean("langSelected", true);
                        editor.apply();
                        LocaleUtils.updateConfig(mContext,lang);
                        Intent intent = mContext.getIntent();
                        mContext.overridePendingTransition(0, 0);
                        mContext.finish();
                        intent.addFlags(Intent.FLAG_ACTIVITY_NO_ANIMATION);
                        intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
                        mContext.overridePendingTransition(0, 0);
                        mContext.startActivity(intent);

最后在Splash Activity或Launching Activity中粘贴以下代码。

SharedPreferences preferences = PreferenceManager.getDefaultSharedPreferences(this);
    String lang = preferences.getString("lang", "");
    boolean langSelected = preferences.getBoolean("langSelected", false);
    SharedPreferences.Editor editor = preferences.edit();
    if (langSelected) {
        editor.clear();
        editor.putString("lang", lang);
        editor.putBoolean("langSelected", true);
        editor.apply();
        LocaleUtils.updateConfig(this,lang);
    } else {
        LocaleUtils.updateConfig(this, Locale.getDefault().getLanguage());
        editor.clear();
        editor.putString("lang", Locale.getDefault().getLanguage());
        editor.putBoolean("langSelected", false);
        editor.apply();
    }

答案 6 :(得分:0)

您需要使用getApplicationContext()代替getContext()

答案 7 :(得分:0)

在所有来源中使用所有解决方案后,终于找到了我的问题。那让我生气了两天。

每个人都知道在Android Oreo(API 26)中必须使用createConfigurationContext,但是我的问题是将Country名称与local一起使用。

替换

zh_CN en

ar_AE ar

fa_IR fa

我的问题解决了。

希望能帮助别人