为什么我的应用会从资源中获取错误的字符串以进行本地化?

时间:2019-05-29 12:38:16

标签: android

我已经创建了三个用于本地化的文件:"en""ru""uk",我发现问题应用程序可以正确输入某些字符串,而错误输入某些字符串。这是什么意思?例如,一种布局在其工作期间可以包含多个语言环境,而不是一个语言环境,这是错误的。我看到一些用户通过向buil.gradle添加一些代码行来解决了这个问题,但是也许我们有另一种解决方案,为什么几周前我的项目使用Java时它能很好地工作。我敢肯定,不可能是从一种语言切换到另一种语言造成的。在recyclerView上,我可以看到类似的问题。在活动中,我可以这样设置当前语言:

sp = this.getSharedPreferences("app_data", 0)
        val lang = sp!!.getString("language", Locale.getDefault().language)
        val locale = Locale(lang)
        Locale.setDefault(locale)
        val config = Configuration()
        config.setLocale(locale)

        resources.updateConfiguration(
                config, resources.displayMetrics
        )

但是不赞成使用updateConfiguration,因此我寻找了一些可以使用的新变体。然后我发现了this问题,但这并没有帮助我。我需要在RV项目,活动和片段上设置语言,但是我做不到。为什么会发生,如何解决这个问题?

2 个答案:

答案 0 :(得分:2)

更改语言后,您必须刷新活动:请查看演示代码以选择语言。希望这会对您有所帮助!

public class ChangeLangDemo extends AppCompatActivity {
        private static final String TAG = ChangeLangDemo.class.getSimpleName();
        private String[] languages = {"Select Language", "English", "Hindi", "Gujarati"};
        private TextView textView;

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

            AppCompatSpinner spinner = findViewById(R.id.language_spinner);
            textView = findViewById(R.id.text);

            loadLocale();

            ArrayAdapter adapter = new ArrayAdapter<>(this, android.R.layout.simple_spinner_dropdown_item, languages);
            spinner.setAdapter(adapter);


            spinner.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() {
                @Override
                public void onItemSelected(AdapterView<?> parent, View view, int position, long id) {

                    String lang = "en";
                    switch (position) {
                        case 0:
                            break;
                        case 1:
                            lang = "en";
                            break;
                        case 2:
                            lang = "hi";
                            break;
                        case 3:
                            lang = "gu";
                            break;
                        default:
                            break;
                    }

                    changeLang(lang);
                }

                @Override
                public void onNothingSelected(AdapterView<?> parent) {

                }

            });

        }


        public void saveLocale(String lang) {
            String langPref = "Language";
            SharedPreferences prefs = getSharedPreferences("CommonPrefs", Activity.MODE_PRIVATE);
            SharedPreferences.Editor editor = prefs.edit();
            editor.putString(langPref, lang);
            editor.apply();
        }

        public void loadLocale() {
            String langPref = "Language";
            SharedPreferences prefs = getSharedPreferences("CommonPrefs", Activity.MODE_PRIVATE);
            String language = prefs.getString(langPref, "");
            changeLang(language);
        }

        public void changeLang(String lang) {
            if (lang.equalsIgnoreCase(""))
                return;
            Locale locale = new Locale(lang);
            saveLocale(lang);
            Locale.setDefault(locale);
            android.content.res.Configuration config = new android.content.res.Configuration();
            config.locale = locale;
            getBaseContext().getResources().updateConfiguration(config, getBaseContext().getResources().getDisplayMetrics());
            updateTexts();
        }

        private void updateTexts() {
            textView.setText(getResources().getText(R.string.text));
        }
    }

答案 1 :(得分:1)

我实际上只需要自己做。这是一个痛苦,特别是对于较新的设备。

首先,您需要一个语言环境帮助器:

package za.co.overtake.onlinetrucks.utils;

import android.annotation.TargetApi;
import android.content.Context;
import android.content.SharedPreferences;
import android.content.res.Configuration;
import android.content.res.Resources;
import android.os.Build;
import android.preference.PreferenceManager;

import java.util.Locale;

/**
 * This class is used to change your application locale and persist this change for the next time
 * that your app is going to be used.
 * <p/>
 * You can also change the locale of your application on the fly by using the setLocale method.
 * <p/>
 * Created by gunhansancar on 07/10/15.
 */
public class LocaleHelper {

    private static final String SELECTED_LANGUAGE = "Locale.Helper.Selected.Language";

    public static Context onAttach(Context context) {
      String lang = getPersistedData(context, Locale.getDefault().getLanguage());
      return setLocale(context, lang);
    }

    public static Context onAttach(Context context, String defaultLanguage) {
      String lang = getPersistedData(context, defaultLanguage);
      return setLocale(context, lang);
    }

    public static String getLanguage(Context context) {
      return getPersistedData(context, Locale.getDefault().getLanguage());
    }

    public static Context setLocale(Context context, String language) {
      persist(context, language);

      if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
        return updateResources(context, language);
      }

    return updateResourcesLegacy(context, language);
    }

    private static String getPersistedData(Context context, String defaultLanguage) {
    SharedPreferences preferences = PreferenceManager.getDefaultSharedPreferences(context);
    return preferences.getString(SELECTED_LANGUAGE, defaultLanguage);
}

private static void persist(Context context, String language) {
    SharedPreferences preferences = PreferenceManager.getDefaultSharedPreferences(context);
    SharedPreferences.Editor editor = preferences.edit();

    editor.putString(SELECTED_LANGUAGE, language);
    editor.apply();
}

@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);
    configuration.setLayoutDirection(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));
}

这应该处理大多数情况。但是,我使用ViewModels,由于某种原因,当我执行getApplication.getResources()。getString()时,有时会返回错误的字符串。在这种情况下,我制作了一种Utility方法来解决此问题:

public static Resources getResources(Context context) {
    return LocaleHelper.onAttach(context).getResources();
}

所以现在我可以使用此方法代替通常的getResources()。

这是一个真正的痛苦,但这是我可以使翻译在高于android 8和低于android8的两种设备上运行的唯一方法。

我将稍后发布一些链接...

编辑:另外,就像前面提到的答案一样,您需要重新创建更改语言的活动。

链接:https://proandroiddev.com/change-language-programmatically-at-runtime-on-android-5e6bc15c758

https://gunhansancar.com/change-language-programmatically-in-android/