Inspiriert von der Kalligraphie habe ich schließlich einen Kontext-Wrapper erstellt. In meinem Fall muss ich die Systemsprache überschreiben, um meinen App-Benutzern die Möglichkeit zu geben, die App-Sprache zu ändern. Dies kann jedoch mit jeder Logik angepasst werden, die Sie implementieren müssen.
import android.annotation.TargetApi;
import android.content.Context;
import android.content.ContextWrapper;
import android.content.res.Configuration;
import android.os.Build;
import java.util.Locale;
public class MyContextWrapper extends ContextWrapper {
public MyContextWrapper(Context base) {
super(base);
}
@SuppressWarnings("deprecation")
public static ContextWrapper wrap(Context context, String language) {
Configuration config = context.getResources().getConfiguration();
Locale sysLocale = null;
if (Build.VERSION.SDK_INT > Build.VERSION_CODES.N) {
sysLocale = getSystemLocale(config);
} else {
sysLocale = getSystemLocaleLegacy(config);
}
if (!language.equals("") && !sysLocale.getLanguage().equals(language)) {
Locale locale = new Locale(language);
Locale.setDefault(locale);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
setSystemLocale(config, locale);
} else {
setSystemLocaleLegacy(config, locale);
}
}
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
context = context.createConfigurationContext(config);
} else {
context.getResources().updateConfiguration(config, context.getResources().getDisplayMetrics());
}
return new MyContextWrapper(context);
}
@SuppressWarnings("deprecation")
public static Locale getSystemLocaleLegacy(Configuration config){
return config.locale;
}
@TargetApi(Build.VERSION_CODES.N)
public static Locale getSystemLocale(Configuration config){
return config.getLocales().get(0);
}
@SuppressWarnings("deprecation")
public static void setSystemLocaleLegacy(Configuration config, Locale locale){
config.locale = locale;
}
@TargetApi(Build.VERSION_CODES.N)
public static void setSystemLocale(Configuration config, Locale locale){
config.setLocale(locale);
}
}
Um Ihren Wrapper zu injizieren, fügen Sie in jeder Aktivität den folgenden Code hinzu:
@Override
protected void attachBaseContext(Context newBase) {
super.attachBaseContext(MyContextWrapper.wrap(newBase,"fr"));
}
UPDATE 23/09/2020 Wenn Sie
beispielsweise das App-Thema überschreiben, um beispielsweise den Dunkelmodus anzuwenden, wird ContextThemeWrapper die Spracheinstellung aufheben. Fügen Sie daher Ihrer Aktivität den folgenden Code hinzu, um das gewünschte Gebietsschema zurückzusetzen
@Override
public void applyOverrideConfiguration(Configuration overrideConfiguration) {
Locale locale = new Locale("fr");
overrideConfiguration.setLocale(locale);
super.applyOverrideConfiguration(overrideConfiguration);
}
UPDATE 19.10.2008
Manchmal wird das Konfigurationsobjekt nach einer Änderung der Ausrichtung oder einer Pause / Wiederaufnahme der Aktivität auf die Standardsystemkonfiguration zurückgesetzt. Im Ergebnis wird in der App englischer "en" -Text angezeigt, obwohl der Kontext mit dem französischen Gebietsschema "fr" umbrochen wurde . Behalten Sie daher das Context / Activity-Objekt in einer globalen Variablen in Aktivitäten oder Fragmenten niemals als bewährte Methode bei.
Erstellen und verwenden Sie außerdem Folgendes in MyBaseFragment oder MyBaseActivity:
public Context getMyContext(){
return MyContextWrapper.wrap(getContext(),"fr");
}
Diese Vorgehensweise bietet Ihnen eine 100% fehlerfreie Lösung.