Injecting custom IStartupLocaleSelector

How does one inject a custom IStartupLocaleProvider? I have my own game options implementation and would like to read from it at startup.

You would inherit from ILocalesProvider and implement the interface, also mark your class Serializable.
This should work fine in playmode however the editor does not currently support Custom Locale Providers and expects Addressable assets so you won’t see your locales in the Editor unless they are also addressable.
Ill make a note to support them in the editor in the future.

Heres an example GitHub · Enterprise

Also worth noting that there is a bug with SerializeReference that causes classes that do not have any serializable data to not be deserialized. So if your custom locales provider contains no serializable fields then it will be lost after saving. A workaround is to just add some dummy data, this is what we do now. A fix is one the way in 2019.3.

 // There is a bug with SerializeField that causes empty instances to not deserialize. This is a workaround while we wait for the fix (case 1183543)
[SerializeField, HideInInspector]
int dummyObject;

Sorry, I think I made a typo. I meant IStartupLocaleSelector. I have a stored reference to a language code in my own game options storage, and I want to use that to select an available locale.

That said, what’s the canonical way to provide a new external provider (say, reading and caching dynamic localization assets via web service)? ILocaleProvider is not asynchronous.

Oh sorry, I think I misread.
You would do the same thing really but inherit from IStartupLocaleSelector.
Unity will detect any classes that implement IStartupLocaleSelector and they will appear in the list when you press the + button. You can then add it and set its priority in the list, top is the highest, and bottom is the lowest.
For example, the EditorPrefs one does this:

using System;

namespace UnityEngine.Localization.Settings
{
    [Serializable]
    public class PlayerPrefLocaleSelector : IStartupLocaleSelector, IInitialize
    {
        [SerializeField]
        string m_PlayerPreferenceKey = "selected-locale";

        public string PlayerPreferenceKey
        {
            get => m_PlayerPreferenceKey;
            set => m_PlayerPreferenceKey = value;
        }

        public void PostInitialization(LocalizationSettings settings)
        {
            if (Application.isPlaying)
            {
                // Record the new selected locale so it can persist between runs
                var selectedLocale = settings.GetSelectedLocale();
                if (selectedLocale != null)
                    PlayerPrefs.SetString(PlayerPreferenceKey, selectedLocale.Identifier.Code);
            }
        }

        public Locale GetStartupLocale(ILocalesProvider availableLocales)
        {
            if (PlayerPrefs.HasKey(PlayerPreferenceKey))
            {
                var code = PlayerPrefs.GetString(PlayerPreferenceKey);
                if (!string.IsNullOrEmpty(code))
                {
                    return availableLocales.GetLocale(code);
                }
            }

            // No locale could be found.
            return null;
        }
    }
}

If you add the IPreloadRequired interface to your ILocalesProvider then it will let you do asynchronous initialization during the LocalizationSettings initialization operation.

Excellent. It’s amazing to see how thoroughly was package was thought out. Definitely would like to see some of this added to the documentation, since I don’t think I’m the only one with workflows like this.

Thanks :slight_smile:
The documentation is quite light at the moment, we will add lots to it before we go out of preview.