Inspirado na Caligrafia , acabei criando um wrapper de contexto. No meu caso, preciso substituir o idioma do sistema para fornecer aos usuários do meu aplicativo a opção de alterar o idioma do aplicativo, mas isso pode ser personalizado com qualquer lógica que você precise implementar.
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);
}
}
e para injetar seu wrapper, em cada atividade adicione o seguinte código:
@Override
protected void attachBaseContext(Context newBase) {
super.attachBaseContext(MyContextWrapper.wrap(newBase,"fr"));
}
ATUALIZAÇÃO 22/12/2020
Após a implementação da biblioteca de materiais android do ContextThemeWrapper para suportar o modo escuro, a configuração de idioma seria interrompida e a configuração de idioma seria perdida. Após meses de coçar a cabeça, o problema foi resolvido adicionando o seguinte código ao método Activity and Fragment onCreate
Context context = MyContextWrapper.wrap(this, "fr");
getResources().updateConfiguration(context.getResources().getConfiguration(), context.getResources().getDisplayMetrics());
ATUALIZAÇÃO 19/10/2018
Às vezes, após a mudança de orientação ou pausa / retomada da atividade, o objeto Configuração é redefinido para a configuração padrão do sistema e, como resultado, veremos o aplicativo exibindo o texto "en" em inglês, embora tenhamos agrupado o contexto com a localidade "fr" em francês . Portanto, como uma boa prática, nunca retenha o objeto Contexto / Atividade em uma variável global em atividades ou fragmentos.
além disso, crie e use o seguinte em um MyBaseFragment ou MyBaseActivity:
public Context getMyContext(){
return MyContextWrapper.wrap(getContext(),"fr");
}
Esta prática irá fornecer a você uma solução 100% livre de bugs.