modular-avatar/Editor/Localization/Localization.cs

142 lines
4.4 KiB
C#
Raw Normal View History

using System;
2022-10-17 05:05:26 +08:00
using System.Collections.Generic;
using System.Collections.Immutable;
using System.IO;
using System.Linq;
using nadena.dev.ndmf.localization;
using nadena.dev.ndmf.ui;
2022-10-17 05:05:26 +08:00
using Newtonsoft.Json;
using UnityEditor;
using UnityEngine;
2022-11-11 12:39:58 +08:00
namespace nadena.dev.modular_avatar.core.editor
2022-10-17 05:05:26 +08:00
{
internal static class Localization
2022-10-17 05:05:26 +08:00
{
2023-01-19 20:32:44 +08:00
public static event Action OnLangChange;
private const string FallbackLanguage = "en";
2022-10-17 05:05:26 +08:00
private const string localizationPathGuid = "488c994003974b3ab2796371cf627bca";
private static string localizationPathRoot = AssetDatabase.GUIDToAssetPath(localizationPathGuid);
private static ImmutableDictionary<string, string> SupportedLanguageDisplayNames
= ImmutableDictionary<string, string>.Empty
.Add("en-us", "English")
.Add("ja-jp", "日本語")
.Add("zh-hans", "简体中文")
.Add("ko-kr", "한국어");
2022-10-17 05:05:26 +08:00
2023-01-19 20:32:44 +08:00
private static ImmutableList<string>
SupportedLanguages = new string[] {"en-us", "ja-jp", "zh-hans", "ko-kr"}.ToImmutableList();
2022-10-17 05:05:26 +08:00
private static string[] DisplayNames = SupportedLanguages.Select(l =>
{
return SupportedLanguageDisplayNames.TryGetValue(l, out var displayName) ? displayName : l;
}).ToArray();
private static Dictionary<string, ImmutableDictionary<string, string>> Cache
= new Dictionary<string, ImmutableDictionary<string, string>>();
internal static string OverrideLanguage { get; set; } = null;
public static Localizer L { get; private set; }
2022-10-17 05:05:26 +08:00
static Localization()
2022-10-17 05:05:26 +08:00
{
Localizer localizer = new Localizer(SupportedLanguages[0], () =>
2022-10-17 05:05:26 +08:00
{
List<(string, Func<string, string>)> languages = new List<(string, Func<string, string>)>();
foreach (var lang in SupportedLanguages)
{
languages.Add((lang, LanguageLookup(lang)));
}
2022-10-17 05:05:26 +08:00
return languages;
});
L = localizer;
LanguagePrefs.RegisterLanguageChangeCallback(typeof(Localization), _ => OnLangChange?.Invoke());
}
private static Func<string,string> LanguageLookup(string lang)
{
2022-10-17 05:05:26 +08:00
var filename = localizationPathRoot + "/" + lang + ".json";
try
{
var langData = File.ReadAllText(filename);
var langMap = JsonConvert.DeserializeObject<Dictionary<string, string>>(langData);
// return langMap.GetValueOrDefault; - Unity 2019 doesn't have this extension method
return key =>
{
if (langMap.TryGetValue(key, out var val)) return val;
else return null;
};
2022-10-17 05:05:26 +08:00
}
catch (Exception e)
{
Debug.LogError("Failed to load language file " + filename);
Debug.LogException(e);
return (k) => null;
2022-10-17 05:05:26 +08:00
}
}
[MenuItem("Tools/Modular Avatar/Reload localizations")]
public static void Reload()
{
Localizer.ReloadLocalizations();
Cache.Clear();
}
2022-10-17 05:05:26 +08:00
public static GUIContent G(string key)
{
var tooltip = S(key + ".tooltip", null);
return tooltip != null ? new GUIContent(S(key), tooltip) : new GUIContent(S(key));
}
public static string S(string key)
{
return S(key, key);
}
public static string S_f(string key, params string[] format)
{
try
{
return string.Format(S(key, key), format);
}
catch (FormatException)
{
return S(key, key) + "(" + string.Join(", ", format) + ")";
}
}
2022-10-17 05:05:26 +08:00
public static string S(string key, string defValue)
{
if (L.TryGetLocalizedString(key, out var val))
2022-10-17 05:05:26 +08:00
{
return val;
2022-10-17 05:05:26 +08:00
}
else
{
return defValue;
}
}
public static string GetSelectedLocalization()
2022-10-17 05:05:26 +08:00
{
return LanguagePrefs.Language;
2022-10-17 05:05:26 +08:00
}
public static void ShowLanguageUI()
{
EditorGUILayout.Separator();
LanguageSwitcher.DrawImmediate();
2022-10-17 05:05:26 +08:00
}
}
}