0

我能够使用本文中介绍的方法在运行时以编程方式更改本地化。我能够更改应用程序的语言。+ 布局方向 + 工具栏语言。

这是我的代码:

构建.gradle

 buildTypes {
        debug {
            pseudoLocalesEnabled true
        }
dependencies {
    implementation fileTree(dir: 'libs', include: ['*.jar'])
    implementation 'androidx.appcompat:appcompat:1.1.0'
    implementation 'androidx.constraintlayout:constraintlayout:1.1.3'
    implementation 'com.jakewharton:process-phoenix:2.0.0'
}

public class App extends Application {

    @Override
    protected void attachBaseContext(Context base) {
        super.attachBaseContext(LocaleManager.setLocale(base));
    }

    @Override
    public void onConfigurationChanged(Configuration newConfig) {
        super.onConfigurationChanged(newConfig);
        LocaleManager.setLocale(this);
        if (Build.VERSION.SDK_INT >= 26) {
            ProcessPhoenix.triggerRebirth(this); //An activity must have a category of Default in Manifest!
        }
    }
}

public class BaseActivity extends AppCompatActivity {

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        if(Build.VERSION.SDK_INT>=26) {
            LocaleManager.setLocale(this);
        }
        startActivity(new Intent(this,MainActivity.class));
        finish();
    }
}

public class LocaleManager {
private static final String SELECTED_LANGUAGE = "Locale.Helper.Selected.Language";
private static final String DEFAULT_LANGUAGE = "en";

public static Context setLocale(Context c) {
    return setNewLocale(c, getLanguage(c));
}

public static Context setNewLocale(Context c, String language) {
    persistLanguage(c, language);
    return updateResources(c, language);
}

public static String getLanguage(Context c) {
    SharedPreferences preferences = PreferenceManager.getDefaultSharedPreferences(c);
    return preferences.getString(SELECTED_LANGUAGE, DEFAULT_LANGUAGE);
}

public static void persistLanguage(Context c, String language) {
    SharedPreferences preferences = PreferenceManager.getDefaultSharedPreferences(c);
    SharedPreferences.Editor editor = preferences.edit();
    editor.putString(SELECTED_LANGUAGE, language);
    editor.commit(); 
}

private static Context updateResources(Context context, String language) {
    Locale locale = new Locale(language);
    Locale.setDefault(locale);

    Resources res = context.getResources();
    Configuration config = new Configuration(res.getConfiguration());

    if(Build.VERSION.SDK_INT >= 17) {
        config.setLocale(locale);
        res.updateConfiguration(config, res.getDisplayMetrics());
    }
    return context;
}

}


public class MainActivity extends AppCompatActivity {

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
}

//Change English to arabic when user clicked the button.
public void changeLanguage(View view) {
    LocaleManager.persistLanguage(this ,"ar");
    LocaleManager.setLocale(this);
    ProcessPhoenix.triggerRebirth(this); //An activity must have a category of Default in Manifest!

问题: 在我迁移到 androidX 之前,这段代码一直运行良好。现在,它仅适用于装有 Android Oreo+ 的设备。我在迁移到 androidX 之前检查了一个备份版本,它运行良好。我在这里缺少什么吗?

4

2 回答 2

0

更改implementation of 'androidx.appcompat:appcompat:1.1.0'为版本 1.0.0

并保持'androidx.constraintlayout:constraintlayout:1.1.3'版本 1.1.3

解决问题。

此外,如果您使用 implementation 'com.google.android.material:material:1.1.0',请将其更改为版本 1.0.0

于 2020-05-13T11:18:26.503 回答
0

使用 SDK 30 (2020) 的运行时本地化完整解决方案

  • 这在从 OS 版本 19 到 30 的模拟器上进行了测试。它解决了最近报告的有关 OS Marshmallow + Nougat 问题并使用最新的 appcompat + material + constrainlayout 依赖项的问题。
  • 它还处理 RTL 支持的布局方向。

脚步:

  1. 使用适当的限定符为您希望应用支持的不同语言创建字符串 XML 文件。就我而言,我使用英语和阿拉伯语限定词来测试布局 RTL。

  2. pseudoLocalesEnabled true在调试和发布中的 android->build types 下的模块 build.gradle 中添加这一行。

  3. 添加此依赖项以便能够轻松地重新启动您的应用程序:

    '实现'com.jakewharton:process-phoenix:2.0.0'

  4. 创建以下“LocaleManager”java 类来处理将所选语言环境保存到共享首选项并应用必要的运行时语言环境更改。

    public class LocaleManager {
    private static final String SELECTED_LANGUAGE = "Locale.Helper.Selected.Language";
    private static final String DEFAULT_LANGUAGE = "en";
    
    public static Context setLocale(Context c) {
        return setNewLocale(c, getLanguage(c));
    }
    
    public static Context setNewLocale(Context c, String language) {
        persistLanguage(c, language);
        return updateResources(c, language);
    }
    
    public static String getLanguage(Context c) {
        SharedPreferences preferences = PreferenceManager.getDefaultSharedPreferences(c);
        return preferences.getString(SELECTED_LANGUAGE, DEFAULT_LANGUAGE);
    }
    
    public static void persistLanguage(Context c, String language) {
        SharedPreferences preferences = PreferenceManager.getDefaultSharedPreferences(c);
        SharedPreferences.Editor editor = preferences.edit();
        editor.putString(SELECTED_LANGUAGE, language);
        editor.commit(); //U must use Commit and not apply, to avoid opening a new thread, causing a delayed writting in a separate thread!
    }
    
    private static Context updateResources(Context context, String language) {
        Locale locale = new Locale(language);
        Locale.setDefault(locale);
    
        Resources res = context.getResources();
        Configuration config = new Configuration(res.getConfiguration());
    
        if(Build.VERSION.SDK_INT >= 17) {
            config.setLocale(locale);
            res.updateConfiguration(config, res.getDisplayMetrics());
        }
        return context;
    }
    
    }
    
  5. 创建以下扩展应用程序的“应用程序”类。确保您的 MainActivity 或启动器 Activity 在 Manifest 中有一个 Default 类别。

    public class App extends Application {
    
    @Override
    protected void attachBaseContext(Context base) {
        super.attachBaseContext(LocaleManager.setLocale(base));
        Log.e("lang", LocaleManager.getLanguage(base));
    }
    
    //Handles screen rotation only up till API 28 pie
    @Override
    public void onConfigurationChanged(Configuration newConfig) {
        super.onConfigurationChanged(newConfig);
        LocaleManager.setLocale(this);
        if (Build.VERSION.SDK_INT >= 26) {
            ProcessPhoenix.triggerRebirth(this); //An activity must have a category of Default in Manifest!
        }
    }
    }
    

显现

<activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.DEFAULT" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>

<application
        android:name=".App"
        
  1. 创建一个按钮来切换 MainActivity 中的语言,如下所示:

    public class MainActivity extends AppCompatActivity {
    
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }
    
    //Change English to arabic when user clicked the button.
    public void changeLanguageToArabic(View view) {
        LocaleManager.persistLanguage(this, "ar");
        LocaleManager.setLocale(this);
        ProcessPhoenix.triggerRebirth(this); //An activity must have a category of Default in Manifest!
    }
    
    
    public void goToSecondActivity(View view) {
        Intent intent = new Intent(this, MainActivity2.class);
        startActivity(intent);
    }
    

您选择的语言环境将在活动和重新启动您的应用程序时保持不变。

于 2020-11-30T13:03:25.267 回答