Почему мое приложение получает неправильные строки из ресурсов для локализации?

Я создал три файла для локализации: "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 устарел, и я искал несколько новых вариантов, которые будут работать. Затем я нашел вопрос это, но он мне не помог. Мне нужно установить язык для элементов, действий и фрагментов RV, но я не могу этого сделать. Почему это происходит и как я могу решить эту проблему?

Вы не должны использовать общие настройки для хранения языковых настроек. Пусть система справится с этим, пользователь может настроить свои языки в настройках.

Tim 29.05.2019 14:41

@TimCastelijns, но пользователь не может понять системный язык или ему будет проще использовать родной язык в моем приложении, что, если он изменит язык в приложении без связи с системным языком?

Andrew 29.05.2019 14:43
4
2
1 438
2
Перейти к ответу Данный вопрос помечен как решенный

Ответы 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));
        }
    }
Ответ принят как подходящий

Я на самом деле ПРОСТО должен был сделать это сам. Это боль, особенно для новых устройств.

Во-первых, вам нужен помощник по локали:

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(), он иногда возвращает неправильную строку. В таком случае я сделал служебный метод, чтобы решить эту проблему:

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/

Другие вопросы по теме