当用户更改语言时,我执行以下代码,它适用于活动中的当前片段,但是如果我转到其他片段,它会部分更新语言,一些字符串会更新并显示旧语言,最重要的是日期不会内部碎片和其他活动的变化。
我在牛轧糖,棉花糖和奥利奥中进行了测试,并在整个操作系统中进行了测试。
当用户更改我执行的语言时。
LocaleHelper.setLocale(getApplicationContext(), language);
recreate();
LocalHelper
public static Context setLocale(Context context, String language) {
persist(context, language);
Log.d("LocaleSet", language);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
return updateResources(context, language);
}
return updateResourcesLegacy(context, language);
}
后棉花糖OS的方法。
@TargetApi(Build.VERSION_CODES.N)
private static Context updateResources(Context context, String language) {
Locale locale = new Locale(language);
Locale.setDefault(locale);
Configuration configuration = context.getResources().getConfiguration();
configuration.setLocale(locale);
return context.createConfigurationContext(configuration);
}
对于牛轧糖
@SuppressWarnings("deprecation")
private static Context updateResourcesLegacy(Context context, String language) {
Locale locale = new Locale(language);
Locale.setDefault(locale);
Resources resources = context.getResources();
Configuration configuration = resources.getConfiguration();
configuration.locale = locale;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1) {
configuration.setLayoutDirection(locale);
}
resources.updateConfiguration(configuration, resources.getDisplayMetrics());
return context;
}
在每个活动中,我执行以下代码。
@Override
protected void attachBaseContext(Context base) {
super.attachBaseContext(LocaleHelper.onAttach(base, LocaleHelper.getLanguage(base)));
}
Mainfest
<application
android:name=".GlobalApplication"
android:allowBackup="false"
android:icon="@mipmap/app_icon"
android:label="@string/app_name"
android:roundIcon="@mipmap/app_icon"
android:supportsRtl="true"
android:theme="@style/AppTheme"
android:windowSoftInputMode="adjustPan"
tools:replace="android:allowBackup">
<activity
android:name=".activities.homeactivity.HomeActivity"
android:label="@string/app_name"
android:screenOrientation="portrait"
android:configChanges="locale"
android:theme="@style/AppTheme.NoActionBar" />
<activity
android:name=".activities.profilepageactivity.ProfilePageActivity"
android:label="@string/title_activity_profile_page"
android:screenOrientation="portrait"
android:theme="@style/AppTheme.NoActionBar" />
我假设你正在使用this方法。如果没有建议对AndroidManifest.xml
进行任何更改,那么android:configChanges="locale"
可能会导致您定义的错误行为。
对于日期格式化,您应该考虑到您的应用程序不使用Locale.getDefault()
,而是由用户和您的LocaleHelper
机制定义的不同的东西。
关于配置更改的一些额外细节
android:configChanges
表示您不希望系统在其中一个提供的属性发生时重新创建您的活动。在你的情况下locale
。
在使用该方法开发方面,为了在清单中正确处理此选项,您必须实现
@Override
public void onConfigurationChanged(Configuration newConfig) {
// refresh your views here
super.onConfigurationChanged(newConfig);
}
然后执行自己的处理。在你的情况下你不需要的东西。
在清单文件中扩展的应用程序类。
package com.example.languageapplication;
import android.content.Context;
public class Application extends android.app.Application {
private static Application applicationInstance;
public static synchronized Application getInstance() {
return applicationInstance;
}
@Override
public void onCreate() {
super.onCreate();
applicationInstance = this;
}
public void initAppLanguage(Context context) {
LocaleUtils.initialize(context, LocaleUtils.getSelectedLanguageId());
}
}
Locale utils类,用于共享首选项中的set语言,以及set语言的更新配置
public class LocaleUtils {
public static final String ENGLISH = "en";
public static final String FRENCH = "fr";
public static final String SPANISH = "es";
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;
}
@Retention(RetentionPolicy.SOURCE)
@StringDef({ENGLISH, FRENCH, SPANISH})
public @interface LocaleDef {
String[] SUPPORTED_LOCALES = {ENGLISH, FRENCH, SPANISH};
}
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");
}
}
单击另一个活动中的按钮设置语言并重新启动启动器活动
public class setLaunguageActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main2);
}
public void setEnglish(View view) {
LocaleUtils.setSelectedLanguageId("en");
Intent i = getBaseContext().getPackageManager().getLaunchIntentForPackage(getBaseContext().getPackageName());
startActivity(i);
}
public void setFrance(View view) {
LocaleUtils.setSelectedLanguageId("fr");
Intent i = getBaseContext().getPackageManager().getLaunchIntentForPackage(getBaseContext().getPackageName());
startActivity(i);
}
}
像这样在主活动中初始化Application类
public class MainActivity extends AppCompatActivity {
TextView textInfo;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
Application.getInstance().initAppLanguage(this);
setContentView(R.layout.activity_main);
}
这段代码对我有用