using System; using System.Collections; using System.Collections.Generic; using System.Collections.Specialized; using System.ComponentModel; using System.Diagnostics; using System.Globalization; using System.IO; using System.Linq; using System.Reflection; using System.Reflection.Emit; using System.Runtime.CompilerServices; using System.Runtime.InteropServices; using System.Runtime.Versioning; using System.Text; using System.Text.RegularExpressions; using BepInEx; using BepInEx.Configuration; using BepInEx.Logging; using GUIFramework; using HarmonyLib; using LitJson; using ModUtils; using TMPro; using UnityEngine; [assembly: CompilationRelaxations(8)] [assembly: RuntimeCompatibility(WrapNonExceptionThrows = true)] [assembly: Debuggable(DebuggableAttribute.DebuggingModes.IgnoreSymbolStoreSequencePoints)] [assembly: AssemblyTitle("BetterPortal")] [assembly: AssemblyDescription("")] [assembly: AssemblyConfiguration("")] [assembly: AssemblyCompany("")] [assembly: AssemblyProduct("BetterPortal")] [assembly: AssemblyCopyright("Copyright © 2022-2026")] [assembly: AssemblyTrademark("")] [assembly: ComVisible(false)] [assembly: Guid("AF8F4729-28E8-452D-8010-40C508E79EB0")] [assembly: AssemblyFileVersion("1.0.7.0")] [assembly: TargetFramework(".NETFramework,Version=v4.8", FrameworkDisplayName = "")] [assembly: AssemblyVersion("1.0.7.0")] namespace ModUtils { public class Configuration { private sealed class LocalizedConfigEntry { public ConfigEntryBase Entry { get; set; } public L10N Localization { get; set; } public string Section { get; set; } public string Key { get; set; } public ConfigurationManagerAttributes Attributes { get; set; } public bool CategoryManaged { get; set; } public bool DispNameManaged { get; set; } public bool DescriptionManaged { get; set; } } private const int DefaultOrder = 4096; private static readonly object LocalizedEntriesLock; private static readonly Dictionary LocalizedEntries; private static readonly string[] DescriptionFieldNames; private readonly ConfigFile _config; private readonly L10N _localization; private Logger _logger; private string Section { get; set; } = "general"; private int Order { get; set; } = 4096; static Configuration() { //IL_005a: Unknown result type (might be due to invalid IL or missing references) //IL_005f: Unknown result type (might be due to invalid IL or missing references) //IL_0075: Unknown result type (might be due to invalid IL or missing references) //IL_0090: Expected O, but got Unknown //IL_00ac: Unknown result type (might be due to invalid IL or missing references) //IL_00b1: Unknown result type (might be due to invalid IL or missing references) //IL_00c7: Unknown result type (might be due to invalid IL or missing references) //IL_00e2: Expected O, but got Unknown LocalizedEntriesLock = new object(); DescriptionFieldNames = new string[4] { "k__BackingField", "Description", "description", "_description" }; LocalizedEntries = new Dictionary(); if (!TomlTypeConverter.CanConvert(typeof(StringList))) { TomlTypeConverter.AddConverter(typeof(StringList), new TypeConverter { ConvertToObject = (string str, Type type) => (!string.IsNullOrEmpty(str)) ? new StringList(Csv.ParseLine(str, trimUnquotedFields: true)) : new StringList(), ConvertToString = delegate(object obj, Type type) { StringList source = (StringList)obj; return string.Join(", ", source.Select(Csv.Escape)); } }); } if (!TomlTypeConverter.CanConvert(typeof(KeyboardShortcut))) { TomlTypeConverter.AddConverter(typeof(KeyboardShortcut), new TypeConverter { ConvertToObject = (string str, Type type) => KeyboardShortcut.Deserialize(str), ConvertToString = delegate(object obj, Type type) { //IL_0001: Unknown result type (might be due to invalid IL or missing references) //IL_0006: Unknown result type (might be due to invalid IL or missing references) KeyboardShortcut val = (KeyboardShortcut)obj; return ((KeyboardShortcut)(ref val)).Serialize(); } }); } } public Configuration(ConfigFile config, L10N localization) { _config = config; _localization = localization; } internal static void RefreshAllLocalizedMetadata() { LocalizedConfigEntry[] array; lock (LocalizedEntriesLock) { array = LocalizedEntries.Values.ToArray(); } LocalizedConfigEntry[] array2 = array; for (int i = 0; i < array2.Length; i++) { RefreshLocalizedMetadata(array2[i]); } } private static void RegisterLocalizedEntry(ConfigEntryBase entry, L10N localization, string section, string key, ConfigurationManagerAttributes attributes, bool categoryManaged, bool dispNameManaged, bool descriptionManaged) { LocalizedConfigEntry localizedConfigEntry = new LocalizedConfigEntry { Entry = entry, Localization = localization, Section = section, Key = key, Attributes = attributes, CategoryManaged = categoryManaged, DispNameManaged = dispNameManaged, DescriptionManaged = descriptionManaged }; lock (LocalizedEntriesLock) { LocalizedEntries[entry] = localizedConfigEntry; } RefreshLocalizedMetadata(localizedConfigEntry); } private static void RefreshLocalizedMetadata(LocalizedConfigEntry entry) { //IL_012a: Unknown result type (might be due to invalid IL or missing references) //IL_0131: Expected O, but got Unknown if (entry?.Entry == null || entry.Localization == null) { return; } try { ConfigurationManagerAttributes attributes = entry.Attributes; if (attributes != null) { if (entry.CategoryManaged) { attributes.Category = entry.Localization.Translate("@config_" + entry.Section + "_section"); } if (entry.DispNameManaged) { attributes.DispName = entry.Localization.Translate("@config_" + entry.Section + "_" + entry.Key + "_name"); } string text = (attributes.Description = (entry.DescriptionManaged ? entry.Localization.Translate("@config_" + entry.Section + "_" + entry.Key + "_description") : attributes.Description)); ConfigDescription description = entry.Entry.Description; AcceptableValueBase val = ((description != null) ? description.AcceptableValues : null); object[] array = ReplaceConfigurationManagerAttributes((description != null) ? description.Tags : null, attributes); ConfigDescription description2 = new ConfigDescription(text, val, array); if (!TrySetEntryDescription(entry.Entry, description2)) { Debug.LogError((object)("[ModUtils] Failed to replace ConfigDescription for [" + entry.Section + ":" + entry.Key + "].")); } } } catch (Exception arg) { Debug.LogError((object)$"[ModUtils] Failed to refresh localized config metadata for [{entry.Section}:{entry.Key}]: {arg}"); } } private static object[] ReplaceConfigurationManagerAttributes(object[] tags, ConfigurationManagerAttributes attributes) { if (tags == null || tags.Length == 0) { return new object[1] { attributes }; } bool flag = false; List list = new List(tags.Length); foreach (object obj in tags) { if (obj is ConfigurationManagerAttributes) { if (!flag) { list.Add(attributes); flag = true; } } else { list.Add(obj); } } if (!flag) { list.Add(attributes); } return list.ToArray(); } private static bool TrySetEntryDescription(ConfigEntryBase entry, ConfigDescription description) { MethodInfo methodInfo = AccessTools.PropertySetter(((object)entry).GetType(), "Description") ?? AccessTools.PropertySetter(typeof(ConfigEntryBase), "Description"); if (methodInfo != null) { methodInfo.Invoke(entry, new object[1] { description }); return true; } string[] descriptionFieldNames = DescriptionFieldNames; foreach (string text in descriptionFieldNames) { FieldInfo fieldInfo = AccessTools.Field(((object)entry).GetType(), text) ?? AccessTools.Field(typeof(ConfigEntryBase), text); if (!(fieldInfo == null) && typeof(ConfigDescription).IsAssignableFrom(fieldInfo.FieldType)) { fieldInfo.SetValue(entry, description); return true; } } return false; } private void LogSection(string section) { _logger?.Debug("[CONFIG] === " + GetSection(section) + " / [" + section + "]"); } private void LogConfigEntry(ConfigEntry entry, ConfigurationManagerAttributes attributes) { _logger?.Debug("[CONFIG] ==== " + attributes.DispName + " / [" + ((ConfigEntryBase)entry).Definition.Key + "]"); _logger?.Debug("[CONFIG] " + ((ConfigEntryBase)entry).Description.Description); _logger?.Debug("[CONFIG] "); Type type = typeof(T); object defaultValue = ((ConfigEntryBase)entry).DefaultValue; if (attributes.ObjToStr != null) { _logger?.Debug("[CONFIG] - Default value: " + attributes.ObjToStr(defaultValue)); } else if (TomlTypeConverter.CanConvert(type)) { _logger?.Debug("[CONFIG] - Default value: " + TomlTypeConverter.ConvertToString(defaultValue, type)); } else { _logger?.Debug($"[CONFIG] - Default value: {defaultValue}"); } AcceptableValueBase acceptableValues = ((ConfigEntryBase)entry).Description.AcceptableValues; if (acceptableValues != null) { string[] array = acceptableValues.ToDescriptionString().Split('\n'); foreach (string text in array) { _logger?.Debug("[CONFIG] - " + text); } } else if (type.IsEnum) { List list = (from x in Enum.GetValues(type).OfType() select Enum.GetName(type, x)).ToList(); _logger?.Debug("[CONFIG] - Acceptable values: " + string.Join(", ", list)); if (type.GetCustomAttributes(typeof(FlagsAttribute), inherit: false).Any()) { IEnumerable values = list.Where((string x) => !string.Equals(x, "none", StringComparison.OrdinalIgnoreCase) && !string.Equals(x, "all", StringComparison.OrdinalIgnoreCase)).Take(2); _logger?.Debug("[CONFIG] - Multiple values can be set at the same time by separating them with , (e.g. " + string.Join(", ", values) + ")"); } } _logger?.Debug("[CONFIG] "); } public void SetDebugLogger(Logger logger) { _logger = logger; } public void ChangeSection(string section, int initialOrder = 4096) { Section = section; Order = initialOrder; if (_logger != null) { LogSection(Section); } } private ConfigEntry Bind(string section, int order, string key, T defaultValue, AcceptableValueBase acceptableValue = null, Action initializer = null) { //IL_00b9: Unknown result type (might be due to invalid IL or missing references) //IL_00c3: Expected O, but got Unknown L10N.EnsurePatched(); string section2 = GetSection(section); string name = GetName(section, key); ConfigurationManagerAttributes configurationManagerAttributes = new ConfigurationManagerAttributes { Category = section2, Order = order, DispName = name, CustomDrawer = ConfigurationCustomDrawer.Get(typeof(T), acceptableValue) }; initializer?.Invoke(configurationManagerAttributes); bool categoryManaged = string.Equals(configurationManagerAttributes.Category, section2, StringComparison.Ordinal); bool dispNameManaged = string.Equals(configurationManagerAttributes.DispName, name, StringComparison.Ordinal); bool flag = string.IsNullOrEmpty(configurationManagerAttributes.Description); string text = (configurationManagerAttributes.Description = (flag ? GetDescription(section, key) : configurationManagerAttributes.Description)); ConfigEntry val = _config.Bind(section, key, defaultValue, new ConfigDescription(text, acceptableValue, new object[1] { configurationManagerAttributes })); RegisterLocalizedEntry((ConfigEntryBase)(object)val, _localization, section, key, configurationManagerAttributes, categoryManaged, dispNameManaged, flag); if (_logger != null) { LogConfigEntry(val, configurationManagerAttributes); } return val; } public ConfigEntry Bind(string section, string key, T defaultValue, AcceptableValueBase acceptableValue = null, Action initializer = null) { return Bind(section, Order--, key, defaultValue, acceptableValue, initializer); } public ConfigEntry Bind(string section, string key, T defaultValue, (T, T) acceptableValue, Action initializer = null) where T : IComparable { var (val, val2) = acceptableValue; return Bind(section, key, defaultValue, (AcceptableValueBase)(object)new AcceptableValueRange(val, val2), initializer); } public ConfigEntry Bind(string key, T defaultValue, AcceptableValueBase acceptableValue = null, Action initializer = null) { return Bind(Section, key, defaultValue, acceptableValue, initializer); } public ConfigEntry Bind(string key, T defaultValue, (T, T) acceptableValue, Action initializer = null) where T : IComparable { return Bind(Section, key, defaultValue, acceptableValue, initializer); } private string GetSection(string section) { return _localization.Translate("@config_" + section + "_section"); } private string GetName(string section, string key) { return _localization.Translate("@config_" + section + "_" + key + "_name"); } private string GetDescription(string section, string key) { return _localization.Translate("@config_" + section + "_" + key + "_description"); } } public class StringList : ICollection, IEnumerable, IEnumerable { private readonly HashSet _values; public int Count => _values.Count; public bool IsReadOnly => false; public StringList() { _values = new HashSet(); } public StringList(IEnumerable collection) { _values = new HashSet(collection); } IEnumerator IEnumerable.GetEnumerator() { return GetEnumerator(); } public IEnumerator GetEnumerator() { return _values.GetEnumerator(); } public void Add(string item) { _values.Add(item); } public void Clear() { _values.Clear(); } public bool Contains(string item) { return _values.Contains(item); } public void CopyTo(string[] array, int arrayIndex) { _values.CopyTo(array, arrayIndex); } public bool Remove(string item) { return _values.Remove(item); } public bool TryAdd(string item) { return _values.Add(item); } } public class LocalizedDescriptionAttribute : DescriptionAttribute { private readonly string _prefix; private readonly string _key; public override string Description => L10N.Translate(_prefix, _key); public LocalizedDescriptionAttribute(string prefix, string key) : base(key) { _prefix = prefix; _key = key; } public LocalizedDescriptionAttribute(string key) : this("", key) { } } public class AcceptableValueEnum : AcceptableValueBase where T : Enum { private readonly bool _isFlags; private readonly IList _values; public AcceptableValueEnum(params T[] values) : base(typeof(T)) { _isFlags = ((AcceptableValueBase)this).ValueType.GetCustomAttributes(typeof(FlagsAttribute), inherit: false).Any(); _values = MakeValues(((AcceptableValueBase)this).ValueType, (IReadOnlyCollection)(object)values, _isFlags); } private static IList MakeValues(Type type, IReadOnlyCollection values, bool isFlags) { object collection; if (values.Count != 0) { collection = values; } else { collection = Enum.GetValues(type).OfType(); } List list = new List((IEnumerable)collection); if (!isFlags) { return list; } HashSet hashSet = new HashSet(); foreach (long item in list.Select((T @enum) => Convert.ToInt64(@enum))) { long[] array = hashSet.ToArray(); foreach (long num in array) { hashSet.Add(num | item); } hashSet.Add(item); } return hashSet.Select((long x) => Enum.ToObject(type, x)).Cast().ToList(); } public override object Clamp(object value) { if (!((AcceptableValueBase)this).IsValid(value)) { return _values[0]; } return value; } public override bool IsValid(object value) { if (value is T item) { return _values.Contains(item); } if (!(value is IConvertible)) { return false; } long @long = Convert.ToInt64(value); return _values.Any((T x) => Convert.ToInt64(x) == @long); } public override string ToDescriptionString() { StringBuilder stringBuilder = new StringBuilder(); Type type = typeof(T); List list = (from x in _values where Enum.IsDefined(type, x) select Enum.GetName(type, x)).ToList(); stringBuilder.Append("# Acceptable values: ").Append(string.Join(", ", list)); if (!_isFlags) { return stringBuilder.ToString(); } List list2 = list.Where((string x) => !string.Equals(x, "none", StringComparison.OrdinalIgnoreCase) && !string.Equals(x, "all", StringComparison.OrdinalIgnoreCase)).Take(2).ToList(); if (list2.Count == 2) { stringBuilder.Append('\n').Append("# Multiple values can be set at the same time by separating them with , (e.g. ").Append(string.Join(", ", list2)) .Append(")"); } return stringBuilder.ToString(); } } public static class ConfigurationCustomDrawer { public delegate bool IsMatchConfig(Type type, AcceptableValueBase acceptableValue); public delegate Action CustomDrawerSupplier(); private const string L10NPrefix = "mod_utils"; private const string EnabledKey = "@config_button_enabled"; private const string DisabledKey = "@config_button_disabled"; private const string AddKey = "@config_button_add"; private const string RemoveKey = "@config_button_remove"; private static readonly Dictionary CustomDrawers; private static bool _defaultTranslationsInitialized; private static readonly Dictionary> BuiltinDrawerTranslations; private static readonly Dictionary> CustomDrawerTranslations; private static readonly Dictionary OwnedDefaults; static ConfigurationCustomDrawer() { BuiltinDrawerTranslations = new Dictionary>(StringComparer.OrdinalIgnoreCase) { { "English", new Dictionary { { "@config_button_enabled", "Enabled" }, { "@config_button_disabled", "Disabled" }, { "@config_button_add", "Add" }, { "@config_button_remove", "Remove" } } }, { "Japanese", new Dictionary { { "@config_button_enabled", "有効" }, { "@config_button_disabled", "無効" }, { "@config_button_add", "追加" }, { "@config_button_remove", "削除" } } } }; CustomDrawerTranslations = new Dictionary>(StringComparer.OrdinalIgnoreCase); OwnedDefaults = new Dictionary(); CustomDrawers = new Dictionary { { IsBool, () => Bool }, { IsFloatWithRange, () => FloatSlider }, { IsStringList, StringList }, { IsFlagsEnum, () => Flags } }; } public static void RegisterDefaultDrawerTranslation(string language, string key, string value) { if (!CustomDrawerTranslations.TryGetValue(language, out var value2)) { value2 = new Dictionary(); CustomDrawerTranslations[language] = value2; } value2[key] = value; if (_defaultTranslationsInitialized) { RefreshDefaultTranslations(); } } private static void EnsureDefaultTranslations() { if (!_defaultTranslationsInitialized) { _defaultTranslationsInitialized = TryAddDefaultTranslations(); } } internal static void RefreshDefaultTranslations() { _defaultTranslationsInitialized = false; _defaultTranslationsInitialized = TryAddDefaultTranslations(); if (!_defaultTranslationsInitialized) { Debug.LogWarning((object)"[ModUtils] Failed to refresh custom drawer fallback translations for the current language."); } } private static Dictionary ResolveDrawerTranslations(string language) { Dictionary dictionary = new Dictionary(BuiltinDrawerTranslations["English"]); if (CustomDrawerTranslations.TryGetValue("English", out var value)) { foreach (KeyValuePair item in value) { dictionary[item.Key] = item.Value; } } if (!string.Equals(language, "English", StringComparison.OrdinalIgnoreCase)) { if (BuiltinDrawerTranslations.TryGetValue(language, out var value2)) { foreach (KeyValuePair item2 in value2) { dictionary[item2.Key] = item2.Value; } } if (CustomDrawerTranslations.TryGetValue(language, out var value3)) { foreach (KeyValuePair item3 in value3) { dictionary[item3.Key] = item3.Value; } } } return dictionary; } private static bool TryAddDefaultTranslations() { Localization instance; try { instance = Localization.instance; } catch (Exception) { return false; } if (instance == null) { return false; } Dictionary field = Reflections.GetField>(instance, "m_translations"); if (field == null) { return false; } string text; try { text = instance.GetSelectedLanguage(); } catch (Exception) { return false; } if (string.IsNullOrEmpty(text)) { text = "English"; } foreach (KeyValuePair item in ResolveDrawerTranslations(text)) { string translationKey = L10N.GetTranslationKey("mod_utils", item.Key); if (field.TryGetValue(translationKey, out var value)) { if (OwnedDefaults.TryGetValue(translationKey, out var value2) && value == value2) { field[translationKey] = item.Value; } } else { field[translationKey] = item.Value; } OwnedDefaults[translationKey] = item.Value; } return true; } private static bool IsBool(Type type, AcceptableValueBase acceptableValue) { return type == typeof(bool); } private static bool IsFloatWithRange(Type type, AcceptableValueBase acceptableValue) { if (type == typeof(float)) { return acceptableValue is AcceptableValueRange; } return false; } private static bool IsStringList(Type type, AcceptableValueBase acceptableValue) { return type == typeof(StringList); } private static bool HasFlagsAttribute(Type type) { if (type.IsEnum) { return type.GetCustomAttributes(typeof(FlagsAttribute), inherit: false).Any(); } return false; } private static bool IsFlagsEnum(Type type, AcceptableValueBase acceptableValue) { return HasFlagsAttribute(type); } public static void Bool(ConfigEntryBase entry) { EnsureDefaultTranslations(); bool flag = (bool)entry.BoxedValue; string text = L10N.Translate("mod_utils", flag ? "@config_button_enabled" : "@config_button_disabled"); bool flag2 = GUILayout.Toggle(flag, text, (GUILayoutOption[])(object)new GUILayoutOption[1] { GUILayout.ExpandWidth(true) }); if (flag2 != flag) { entry.BoxedValue = flag2; } } public static void FloatSlider(ConfigEntryBase entry) { AcceptableValueRange val = (AcceptableValueRange)(object)entry.Description.AcceptableValues; float num = (float)entry.BoxedValue; float minValue = val.MinValue; float maxValue = val.MaxValue; float num2 = GUILayout.HorizontalSlider(num, minValue, maxValue, (GUILayoutOption[])(object)new GUILayoutOption[1] { GUILayout.ExpandWidth(true) }); num2 = Mathf.Floor(num2 * 100f) / 100f; if (Math.Abs(num2 - num) > Mathf.Abs(maxValue - minValue) / 1000f) { entry.BoxedValue = num2; } string text = num.ToString("0.00", CultureInfo.InvariantCulture); string text2 = GUILayout.TextField(text, (GUILayoutOption[])(object)new GUILayoutOption[1] { GUILayout.Width(50f) }); if (text2 == text) { return; } try { num2 = (float)Convert.ToDouble(text2, CultureInfo.InvariantCulture); object boxedValue = Convert.ChangeType(((AcceptableValueBase)val).Clamp((object)num2), entry.SettingType, CultureInfo.InvariantCulture); entry.BoxedValue = boxedValue; } catch (FormatException) { } } private static Action StringList() { string inputText = ""; return delegate(ConfigEntryBase entry) { //IL_011b: Unknown result type (might be due to invalid IL or missing references) //IL_0125: Expected O, but got Unknown //IL_0120: Unknown result type (might be due to invalid IL or missing references) //IL_013a: Unknown result type (might be due to invalid IL or missing references) //IL_0144: Expected O, but got Unknown //IL_013f: Unknown result type (might be due to invalid IL or missing references) EnsureDefaultTranslations(); int num = Mathf.Min(Screen.width, 650); int num2 = num - Mathf.RoundToInt((float)num / 2.5f) - 115; string text = L10N.Translate("mod_utils", "@config_button_add"); string text2 = L10N.Translate("mod_utils", "@config_button_remove"); StringList stringList = new StringList((StringList)entry.BoxedValue); GUILayout.BeginVertical((GUILayoutOption[])(object)new GUILayoutOption[1] { GUILayout.MaxWidth((float)num2) }); GUILayout.BeginHorizontal(Array.Empty()); inputText = GUILayout.TextField(inputText, (GUILayoutOption[])(object)new GUILayoutOption[1] { GUILayout.ExpandWidth(true) }); if (GUILayout.Button(text, (GUILayoutOption[])(object)new GUILayoutOption[1] { GUILayout.ExpandWidth(false) }) && !string.IsNullOrEmpty(inputText)) { if (stringList.TryAdd(inputText)) { entry.BoxedValue = stringList; } inputText = ""; } GUILayout.EndHorizontal(); GUILayout.BeginHorizontal(Array.Empty()); double num3 = 0.0; foreach (string item in stringList.ToList()) { int num4 = Mathf.FloorToInt(GUI.skin.label.CalcSize(new GUIContent(item)).x) + Mathf.FloorToInt(GUI.skin.button.CalcSize(new GUIContent(text2)).x); num3 += (double)num4; if (num3 > (double)num2) { GUILayout.EndHorizontal(); num3 = num4; GUILayout.BeginHorizontal(Array.Empty()); } GUILayout.Label(item, (GUILayoutOption[])(object)new GUILayoutOption[1] { GUILayout.ExpandWidth(false) }); if (GUILayout.Button(text2, (GUILayoutOption[])(object)new GUILayoutOption[1] { GUILayout.ExpandWidth(false) }) && stringList.Remove(item)) { entry.BoxedValue = stringList; } } GUILayout.EndHorizontal(); GUILayout.EndVertical(); GUILayout.FlexibleSpace(); }; } public static void Flags(ConfigEntryBase entry) { //IL_00bd: Unknown result type (might be due to invalid IL or missing references) //IL_00c7: Expected O, but got Unknown //IL_00c2: Unknown result type (might be due to invalid IL or missing references) int num = Mathf.Min(Screen.width, 650); int num2 = num - Mathf.RoundToInt((float)num / 2.5f) - 115; Type settingType = entry.SettingType; long num3 = Convert.ToInt64(entry.BoxedValue); AcceptableValueBase acceptableValues = entry.Description.AcceptableValues; GUILayout.BeginVertical((GUILayoutOption[])(object)new GUILayoutOption[1] { GUILayout.MaxWidth((float)num2) }); int num4 = 0; GUILayout.BeginHorizontal(Array.Empty()); foreach (object value2 in Enum.GetValues(settingType)) { if (acceptableValues != null && !acceptableValues.IsValid(value2)) { continue; } long num5 = Convert.ToInt64(value2); if (num5 != 0L) { string enumLabel = GetEnumLabel(settingType, value2); int num6 = Mathf.FloorToInt(GUI.skin.toggle.CalcSize(new GUIContent(enumLabel + "_")).x); num4 += num6; if (num4 > num2) { GUILayout.EndHorizontal(); num4 = num6; GUILayout.BeginHorizontal(Array.Empty()); } GUI.changed = false; bool flag = GUILayout.Toggle((num3 & num5) == num5, enumLabel, (GUILayoutOption[])(object)new GUILayoutOption[1] { GUILayout.ExpandWidth(false) }); if (GUI.changed) { long value = (flag ? (num3 | num5) : (num3 & ~num5)); entry.BoxedValue = Enum.ToObject(settingType, value); } } } GUILayout.EndHorizontal(); GUI.changed = false; GUILayout.EndVertical(); GUILayout.FlexibleSpace(); } public static Action MultiSelect(Func> allElementSupplier, Func labelGenerator) where T : IComparable { return delegate(ConfigEntryBase entry) { //IL_013b: Unknown result type (might be due to invalid IL or missing references) //IL_0145: Expected O, but got Unknown //IL_0140: Unknown result type (might be due to invalid IL or missing references) int num = Mathf.Min(Screen.width, 650); int num2 = num - Mathf.RoundToInt((float)num / 2.5f) - 115; GUILayout.BeginVertical((GUILayoutOption[])(object)new GUILayoutOption[1] { GUILayout.MaxWidth((float)num2) }); Type settingType = entry.SettingType; ConstructorInfo constructor = settingType.GetConstructor(new Type[1] { typeof(IEnumerable) }); if ((object)constructor == null) { GUILayout.BeginHorizontal(Array.Empty()); GUILayout.Label($"{settingType} does not define a constructor that takes IEnumerable<{typeof(T)}> as an argument.", Array.Empty()); GUILayout.EndHorizontal(); GUI.changed = false; GUILayout.EndVertical(); GUILayout.FlexibleSpace(); } else { ICollection collection = (ICollection)constructor.Invoke(new object[1] { entry.BoxedValue }); AcceptableValueBase acceptableValues = entry.Description.AcceptableValues; int num3 = 0; GUILayout.BeginHorizontal(Array.Empty()); foreach (T item in allElementSupplier()) { if (acceptableValues == null || acceptableValues.IsValid((object)item)) { string text = labelGenerator(item) ?? item.ToString(); int num4 = Mathf.FloorToInt(GUI.skin.toggle.CalcSize(new GUIContent(text + "_")).x); num3 += num4; if (num3 > num2) { GUILayout.EndHorizontal(); num3 = num4; GUILayout.BeginHorizontal(Array.Empty()); } GUI.changed = false; bool flag = GUILayout.Toggle(collection.Contains(item), text, (GUILayoutOption[])(object)new GUILayoutOption[1] { GUILayout.ExpandWidth(false) }); if (GUI.changed) { if (flag) { collection.Add(item); } else { collection.Remove(item); } entry.BoxedValue = collection; } } } GUILayout.EndHorizontal(); GUI.changed = false; GUILayout.EndVertical(); GUILayout.FlexibleSpace(); } }; } public static void Register(IsMatchConfig matcher, CustomDrawerSupplier supplier) { CustomDrawers[matcher] = supplier; } public static Action Get(Type type, AcceptableValueBase acceptableValue) { return (from x in CustomDrawers where x.Key(type, acceptableValue) select x.Value()).FirstOrDefault(); } private static string GetEnumLabel(Type type, object @object) { return (type.GetMember(Enum.GetName(type, @object) ?? "").FirstOrDefault()?.GetCustomAttributes(typeof(DescriptionAttribute), inherit: false).OfType().FirstOrDefault())?.Description ?? @object.ToString(); } } public sealed class ConfigurationManagerAttributes { public bool? ShowRangeAsPercent; public Action CustomDrawer; public bool? Browsable; public string Category; public object DefaultValue; public bool? HideDefaultButton; public bool? HideSettingName; public string Description; public string DispName; public int? Order; public bool? ReadOnly; public bool? IsAdvanced; public Func ObjToStr; public Func StrToObj; } public static class Csv { public sealed class Parser { private readonly StringBuilder _fieldBuffer; private readonly List _recordBuffer; private readonly string _source; private readonly bool _trimUnquotedFields; private bool _fieldQuoted; private bool _fieldStarted; private bool _inQuotes; private bool _lastTokenWasDelimiter; private bool _quotedFieldClosed; private int _offset; public Parser(string source, int offset = 0, bool trimUnquotedFields = false) { _source = source ?? ""; _offset = offset; _trimUnquotedFields = trimUnquotedFields; _recordBuffer = new List(); _fieldBuffer = new StringBuilder(); } public List> Parse() { List> list = new List>(); int num = -1; while (HasNext()) { List list2 = ParseLine(); int count = list2.Count; if (count != 0) { if (num == -1) { num = count; } else if (num != count) { throw new Exception("Number of fields in a record is not uniform."); } list.Add(list2); } } return list; } public bool HasNext() { return _offset < _source.Length; } public List ParseLine() { _recordBuffer.Clear(); _fieldBuffer.Clear(); _fieldQuoted = false; _fieldStarted = false; _inQuotes = false; _lastTokenWasDelimiter = false; _quotedFieldClosed = false; List list = new List(); bool recordHasContent = false; while (_offset < _source.Length) { char c = _source[_offset++]; if (!ParseChar(c, ref recordHasContent)) { break; } } if (recordHasContent || _fieldStarted || _lastTokenWasDelimiter) { FlushField(); } if (_recordBuffer.Count > 0) { list.AddRange(_recordBuffer); } return list; } private bool ParseChar(char c, ref bool recordHasContent) { switch (c) { case '"': recordHasContent = true; if (_inQuotes) { if (_offset < _source.Length && _source[_offset] == '"') { _fieldBuffer.Append('"'); _offset++; } else { _inQuotes = false; _quotedFieldClosed = true; } } else if (CanStartQuotedField()) { _fieldQuoted = true; _fieldStarted = true; _inQuotes = true; _quotedFieldClosed = false; _fieldBuffer.Clear(); } else { _fieldBuffer.Append(c); _fieldStarted = true; _quotedFieldClosed = false; } _lastTokenWasDelimiter = false; return true; case ',': if (!_inQuotes) { recordHasContent = true; FlushField(); _lastTokenWasDelimiter = true; return true; } break; } if ((c == '\r' || c == '\n') && !_inQuotes) { if (c == '\r' && _offset < _source.Length && _source[_offset] == '\n') { _offset++; } return false; } if (_trimUnquotedFields && _fieldQuoted && _quotedFieldClosed && char.IsWhiteSpace(c)) { _lastTokenWasDelimiter = false; return true; } recordHasContent = true; _fieldBuffer.Append(c); _fieldStarted = true; _quotedFieldClosed = false; _lastTokenWasDelimiter = false; return true; } private bool CanStartQuotedField() { if (_fieldStarted) { if (_trimUnquotedFields && !_fieldQuoted) { return FieldBufferIsWhitespace(); } return false; } return true; } private bool FieldBufferIsWhitespace() { for (int i = 0; i < _fieldBuffer.Length; i++) { if (!char.IsWhiteSpace(_fieldBuffer[i])) { return false; } } return true; } private void FlushField() { string text = _fieldBuffer.ToString(); _recordBuffer.Add((_trimUnquotedFields && !_fieldQuoted) ? text.Trim() : text); _fieldBuffer.Clear(); _fieldQuoted = false; _fieldStarted = false; _quotedFieldClosed = false; } } private static readonly char[] MustQuoteChars = new char[4] { '"', ',', '\r', '\n' }; public static string Escape(string field) { if (field == null) { return ""; } if (field.Length == 0 || field.IndexOfAny(MustQuoteChars) != -1 || (field.Length > 0 && (char.IsWhiteSpace(field[0]) || char.IsWhiteSpace(field[field.Length - 1])))) { return "\"" + field.Replace("\"", "\"\"") + "\""; } return field; } public static List> Parse(string csv) { return Parse(csv, trimUnquotedFields: false); } public static List> Parse(string csv, bool trimUnquotedFields) { return new Parser(csv, 0, trimUnquotedFields).Parse(); } public static List ParseLine(string line) { return ParseLine(line, trimUnquotedFields: false); } public static List ParseLine(string line, bool trimUnquotedFields) { return new Parser(line, 0, trimUnquotedFields).ParseLine(); } } public class InstanceCache : MonoBehaviour { private static readonly HashSet Cache; private T _instance; public static Action OnCacheAdded { get; set; } public static Action OnCacheRemoved { get; set; } static InstanceCache() { Cache = new HashSet(); } private void Awake() { _instance = GetInstance(); if (_instance != null) { object obj = _instance; Object val = (Object)((obj is Object) ? obj : null); if (val == null || Object.op_Implicit(val)) { Cache.Add(_instance); OnCacheAdded?.Invoke(_instance); return; } } throw new NullReferenceException("Failed to acquire instance."); } private void OnDestroy() { Cache.Remove(_instance); OnCacheRemoved?.Invoke(_instance); _instance = default(T); } protected virtual T GetInstance() { return ((Component)this).GetComponent(); } public static IEnumerable GetAllInstance() { return Cache.ToList(); } public static void Fill(List buffer) { if (buffer == null) { throw new ArgumentNullException("buffer"); } buffer.Clear(); buffer.AddRange(Cache); } } public enum WorldLevelMatchMode { Exact, CurrentOrHigher, Ignore } public static class Inventories { public static float CurrentWorldLevel { get { if (!((Object)(object)Game.instance != (Object)null)) { return 0f; } return Game.m_worldLevel; } } private static bool IsWorldLevelMatch(ItemData data, float worldLevel, WorldLevelMatchMode matchMode) { return matchMode switch { WorldLevelMatchMode.Exact => (float)data.m_worldLevel == worldLevel, WorldLevelMatchMode.CurrentOrHigher => (float)data.m_worldLevel >= CurrentWorldLevel, WorldLevelMatchMode.Ignore => true, _ => (float)data.m_worldLevel == worldLevel, }; } private static bool IsMatchedItem(ItemData data, string name, float worldLevel, int quality, bool isPrefabName, WorldLevelMatchMode matchMode) { if (data == null) { return false; } bool num; if (!isPrefabName) { num = data.m_shared.m_name == name; } else { if (!((Object)(object)data.m_dropPrefab != (Object)null)) { goto IL_0057; } num = ((Object)data.m_dropPrefab).name == name; } if (num && (quality < 0 || data.m_quality == quality)) { return IsWorldLevelMatch(data, worldLevel, matchMode); } goto IL_0057; IL_0057: return false; } private static void NotifyChanged(Inventory inventory) { Reflections.InvokeMethod(inventory, "Changed"); } private static int CountItems(Inventory inventory, string name, float worldLevel, int quality, bool isPrefabName = false, WorldLevelMatchMode matchMode = WorldLevelMatchMode.Exact) { return GetItems(inventory, name, worldLevel, matchMode, quality, isPrefabName).Sum((ItemData x) => x.m_stack); } private static int FindFreeStackSpace(Inventory inventory, string name, float worldLevel, int quality, bool isPrefabName) { return (from item in GetItems(inventory, name, worldLevel, WorldLevelMatchMode.Exact, quality, isPrefabName) where item.m_stack < item.m_shared.m_maxStackSize select item).Sum((ItemData item) => item.m_shared.m_maxStackSize - item.m_stack); } public static IEnumerable GetItems(Inventory inventory, string name, float worldLevel, WorldLevelMatchMode matchMode, int quality = -1, bool isPrefabName = false) { return (from data in inventory.GetAllItems() where IsMatchedItem(data, name, worldLevel, quality, isPrefabName, matchMode) select data).ToList(); } [Obsolete("Use overload with WorldLevelMatchMode parameter")] public static IEnumerable GetItems(Inventory inventory, string name, float worldLevel, int quality = -1, bool isPrefabName = false) { return GetItems(inventory, name, worldLevel, WorldLevelMatchMode.Exact, quality, isPrefabName); } public static int AddItem(Inventory inventory, GameObject prefab, int amount, int quality = -1) { if (amount <= 0) { return 0; } ItemData val = prefab.GetComponent().m_itemData.Clone(); val.m_dropPrefab = prefab; val.m_stack = Mathf.Min(amount, val.m_shared.m_maxStackSize); val.m_quality = Mathf.Clamp(quality, 1, val.m_shared.m_maxQuality); if ((Object)(object)Game.instance != (Object)null) { val.m_worldLevel = Game.m_worldLevel; } int num = CountItems(inventory, val.m_shared.m_name, val.m_worldLevel, val.m_quality); inventory.AddItem(val); return CountItems(inventory, val.m_shared.m_name, val.m_worldLevel, val.m_quality) - num; } public static int FillFreeStackSpace(Inventory from, Inventory to, string name, float worldLevel, int amount, int quality = -1, bool isPrefabName = false) { if (amount <= 0) { return 0; } int num = Mathf.Min(amount, FindFreeStackSpace(to, name, worldLevel, quality, isPrefabName)); if (num == 0) { return 0; } int num2 = 0; IEnumerable items = GetItems(to, name, worldLevel, WorldLevelMatchMode.Exact, quality, isPrefabName); foreach (ItemData item in GetItems(from, name, worldLevel, WorldLevelMatchMode.Exact, quality, isPrefabName)) { foreach (ItemData item2 in items) { if (item2.m_stack >= item2.m_shared.m_maxStackSize) { continue; } int stack = item.m_stack; int num3 = Mathf.Min(Mathf.Min(num, stack), item2.m_shared.m_maxStackSize - item2.m_stack); if (num3 != 0) { item2.m_stack += num3; num2 += num3; num -= num3; from.RemoveItem(item, num3); NotifyChanged(to); if (num == 0) { goto end_IL_00ff; } if (stack - num3 <= 0) { break; } } } continue; end_IL_00ff: break; } return num2; } public static int FillFreeStackSpace(Inventory inventory, string name, float worldLevel, int amount, int quality = -1, bool isPrefabName = false) { if (amount <= 0) { return 0; } int num = amount; int num2 = 0; foreach (ItemData item in GetItems(inventory, name, worldLevel, WorldLevelMatchMode.Exact, quality, isPrefabName)) { if (item.m_stack >= item.m_shared.m_maxStackSize) { continue; } int num3 = Mathf.Min(num, item.m_shared.m_maxStackSize - item.m_stack); if (num3 > 0) { item.m_stack += num3; num2 += num3; num -= num3; NotifyChanged(inventory); if (num == 0) { break; } } } return num2; } public static bool HaveItem(Inventory inventory, string name, float worldLevel, WorldLevelMatchMode matchMode, int amount, int quality = -1, bool isPrefabName = false) { if (amount <= 0) { return true; } int num = 0; foreach (ItemData item in GetItems(inventory, name, worldLevel, matchMode, quality, isPrefabName)) { num += item.m_stack; if (num >= amount) { return true; } } return false; } [Obsolete("Use overload with WorldLevelMatchMode parameter")] public static bool HaveItem(Inventory inventory, string name, float worldLevel, int amount, int quality = -1, bool isPrefabName = false) { return HaveItem(inventory, name, worldLevel, WorldLevelMatchMode.Exact, amount, quality, isPrefabName); } public static int RemoveItem(Inventory inventory, string name, float worldLevel, WorldLevelMatchMode matchMode, int amount, int quality = -1, bool isPrefabName = false) { if (amount <= 0) { return 0; } int num = 0; foreach (ItemData item in GetItems(inventory, name, worldLevel, matchMode, quality, isPrefabName)) { int num2 = Mathf.Min(amount, item.m_stack); inventory.RemoveItem(item, num2); num += num2; amount -= num2; if (amount == 0) { break; } } return num; } [Obsolete("Use overload with WorldLevelMatchMode parameter")] public static int RemoveItem(Inventory inventory, string name, float worldLevel, int amount, int quality = -1, bool isPrefabName = false) { return RemoveItem(inventory, name, worldLevel, WorldLevelMatchMode.Exact, amount, quality, isPrefabName); } } public static class Json { public static void AddImporter(ImporterFunc importer) { JsonMapper.RegisterImporter(importer); } public static void AddExporter(ExporterFunc exporter) { JsonMapper.RegisterExporter(exporter); } public static T Parse(string jsonText, ReaderOption option) { return JsonMapper.ToObject(new JsonReader(jsonText) { AllowComments = option.AllowComments, AllowSingleQuotedStrings = option.AllowSingleQuotedStrings, SkipNonMembers = option.SkipNonMembers }); } public static T Parse(string jsonText) { return Parse(jsonText, new ReaderOption { AllowComments = true, AllowSingleQuotedStrings = false, SkipNonMembers = true }); } public static string ToString(object obj, WriterOption option) { StringBuilder stringBuilder = new StringBuilder(); JsonMapper.ToJson(obj, new JsonWriter(stringBuilder) { IndentValue = option.IndentSize, PrettyPrint = option.Pretty }); return stringBuilder.ToString(); } public static string ToString(object obj) { return ToString(obj, new WriterOption { Pretty = false, IndentSize = 0 }); } } public struct ReaderOption { public bool AllowComments; public bool AllowSingleQuotedStrings; public bool SkipNonMembers; } public struct WriterOption { public bool Pretty; public int IndentSize; } public class L10N { private struct TranslationSource { public L10N Localization; public string Directory; } private static readonly Regex InternalNamePattern; private static readonly Dictionary TranslationCache; private static readonly Dictionary RuntimeWords; private static readonly Regex WordPattern; private const string HarmonyId = "net.eideehi.modutils.localization"; private static bool _patchesApplied; private static bool _initializePatchApplied; private static bool _setLanguagePatchApplied; private static bool _initializeMethodWarningLogged; private static bool _setLanguageMethodWarningLogged; private static bool _localizationCacheWarningLogged; private static string _currentLanguage; private static readonly List _translationSources; private readonly string _prefix; public static event Action LanguageChanged; static L10N() { _translationSources = new List(); InternalNamePattern = new Regex("^(\\$|@)(\\w|\\d|[^\\s(){}[\\]+\\-!?/\\\\&%,.:=<>])+$", RegexOptions.Compiled); TranslationCache = new Dictionary(StringComparer.OrdinalIgnoreCase); RuntimeWords = new Dictionary(StringComparer.OrdinalIgnoreCase); WordPattern = new Regex("(\\$|@)((?:\\w|\\d|[^\\s(){}[\\]+\\-!?/\\\\&%,.:=<>])+)", RegexOptions.Compiled); } public L10N(string prefix) { _prefix = prefix; } internal static void EnsurePatched() { //IL_000e: Unknown result type (might be due to invalid IL or missing references) //IL_0014: Expected O, but got Unknown //IL_0067: Unknown result type (might be due to invalid IL or missing references) //IL_0074: Expected O, but got Unknown //IL_00ce: Unknown result type (might be due to invalid IL or missing references) //IL_00db: Expected O, but got Unknown if (_patchesApplied) { return; } try { Harmony val = new Harmony("net.eideehi.modutils.localization"); if (!_initializePatchApplied) { MethodInfo methodInfo = AccessTools.Method(typeof(Localization), "Initialize", (Type[])null, (Type[])null); if (methodInfo == null) { if (!_initializeMethodWarningLogged) { Debug.LogWarning((object)"[ModUtils] Localization.Initialize method not found. Auto-reload for initialization will not work until it becomes available."); _initializeMethodWarningLogged = true; } } else { val.Patch((MethodBase)methodInfo, (HarmonyMethod)null, new HarmonyMethod(typeof(L10N), "OnLocalizationInitialized", (Type[])null), (HarmonyMethod)null, (HarmonyMethod)null, (HarmonyMethod)null); _initializePatchApplied = true; } } if (!_setLanguagePatchApplied) { MethodInfo methodInfo2 = AccessTools.Method(typeof(Localization), "SetLanguage", (Type[])null, (Type[])null); if (methodInfo2 == null) { if (!_setLanguageMethodWarningLogged) { Debug.LogWarning((object)"[ModUtils] Localization.SetLanguage method not found. Auto-reload for language change will not work until it becomes available."); _setLanguageMethodWarningLogged = true; } } else { val.Patch((MethodBase)methodInfo2, (HarmonyMethod)null, new HarmonyMethod(typeof(L10N), "OnLanguageSet", (Type[])null), (HarmonyMethod)null, (HarmonyMethod)null, (HarmonyMethod)null); _setLanguagePatchApplied = true; } } _patchesApplied = _initializePatchApplied && _setLanguagePatchApplied; } catch (Exception arg) { Debug.LogWarning((object)$"[ModUtils] Failed to patch Localization lifecycle methods. Auto-reload will stay disabled. {arg}"); } } private static void OnLocalizationInitialized() { try { Localization instance = Localization.instance; string text = ((instance != null) ? instance.GetSelectedLanguage() : null); if (!string.IsNullOrEmpty(text)) { HandleLanguageChange(text); } } catch (Exception arg) { Debug.LogError((object)$"[ModUtils] OnLocalizationInitialized error: {arg}"); } } private static void OnLanguageSet(string language) { try { if (!string.IsNullOrEmpty(language)) { HandleLanguageChange(language); } } catch (Exception arg) { Debug.LogError((object)$"[ModUtils] OnLanguageSet error: {arg}"); } } private static void HandleLanguageChange(string language) { ApplyLanguageChange(language, fireLanguageChanged: true); } public static string SyncCurrentLanguage() { EnsurePatched(); try { Localization instance = Localization.instance; if (instance == null) { return null; } string selectedLanguage = instance.GetSelectedLanguage(); if (string.IsNullOrEmpty(selectedLanguage)) { return null; } ApplyLanguageChange(selectedLanguage, fireLanguageChanged: false); return selectedLanguage; } catch (Exception arg) { Debug.LogError((object)$"[ModUtils] SyncCurrentLanguage failed: {arg}"); return null; } } private static void ApplyLanguageChange(string language, bool fireLanguageChanged) { if (!string.IsNullOrEmpty(language) && !string.Equals(language, _currentLanguage, StringComparison.OrdinalIgnoreCase)) { ReloadTranslations(language); _currentLanguage = language; RefreshConfigurationMetadata(); RefreshDefaultDrawerTranslations(); if (fireLanguageChanged) { FireLanguageChanged(language); } } } private static void RefreshConfigurationMetadata() { try { Configuration.RefreshAllLocalizedMetadata(); } catch (Exception arg) { Debug.LogError((object)$"[ModUtils] Failed to refresh localized config metadata: {arg}"); } } private static void RefreshDefaultDrawerTranslations() { try { ConfigurationCustomDrawer.RefreshDefaultTranslations(); } catch (Exception arg) { Debug.LogError((object)$"[ModUtils] Failed to refresh custom drawer translations: {arg}"); } } private static void ReloadTranslations(string language) { TranslationCache.Clear(); foreach (TranslationSource translationSource in _translationSources) { try { new TranslationsLoader(translationSource.Localization).LoadTranslations(translationSource.Directory, language); } catch (Exception arg) { Debug.LogError((object)$"[ModUtils] Failed to reload translations from {translationSource.Directory}: {arg}"); } } Localization localization = TryGetLocalization(); foreach (KeyValuePair runtimeWord in RuntimeWords) { TranslationCache[runtimeWord.Key] = runtimeWord.Value; AddWordToLocalization(localization, runtimeWord.Key, runtimeWord.Value); } } private static void FireLanguageChanged(string language) { Action languageChanged = L10N.LanguageChanged; if (languageChanged == null) { return; } Delegate[] invocationList = languageChanged.GetInvocationList(); foreach (Delegate @delegate in invocationList) { try { ((Action)@delegate)(language); } catch (Exception arg) { Debug.LogError((object)$"[ModUtils] LanguageChanged handler error: {arg}"); } } } public void AddTranslationDirectory(string directory) { EnsurePatched(); _translationSources.Add(new TranslationSource { Localization = this, Directory = directory }); new TranslationsLoader(this).LoadTranslations(directory, DetectCurrentLanguage()); Localization localization = TryGetLocalization(); foreach (KeyValuePair runtimeWord in RuntimeWords) { TranslationCache[runtimeWord.Key] = runtimeWord.Value; AddWordToLocalization(localization, runtimeWord.Key, runtimeWord.Value); } } private static string DetectCurrentLanguage() { if (_currentLanguage != null) { return _currentLanguage; } try { Localization instance = Localization.instance; string text = ((instance != null) ? instance.GetSelectedLanguage() : null); if (!string.IsNullOrEmpty(text)) { return text; } } catch (Exception) { } return "English"; } internal static Localization TryGetLocalization() { try { Localization instance = Localization.instance; if (instance == null) { return null; } return string.IsNullOrEmpty(instance.GetSelectedLanguage()) ? null : instance; } catch (Exception) { return null; } } private static string InvokeTranslate(string word) { Localization val = TryGetLocalization(); TranslationCache.TryGetValue(word, out var value); if (val == null) { return value ?? word; } string text = Reflections.InvokeMethod(val, "Translate", new object[1] { word }); if (text == null || IsMissingTranslation(word, text)) { return value ?? word; } return text; } private static bool IsMissingTranslation(string word, string localized) { if (!string.Equals(localized, word, StringComparison.Ordinal)) { return string.Equals(localized, "[" + word + "]", StringComparison.Ordinal); } return true; } private static string InvokeInsertWordsFallback(string text, IReadOnlyList words) { if (string.IsNullOrEmpty(text)) { return text; } string text2 = text; for (int i = 0; i < words.Count; i++) { text2 = text2.Replace($"${i + 1}", words[i] ?? ""); } return text2; } private static string InvokeInsertWords(string text, string[] words) { if (string.IsNullOrEmpty(text)) { return text; } Localization val = TryGetLocalization(); if (val == null) { return InvokeInsertWordsFallback(text, words); } return Reflections.InvokeMethod(val, "InsertWords", new object[2] { text, words }); } private static void InvokeAddWord(string key, string word) { TranslationCache[key] = word; AddWordToLocalization(TryGetLocalization(), key, word); } private static void InvokeAddRuntimeWord(string key, string word) { RuntimeWords[key] = word; TranslationCache[key] = word; AddWordToLocalization(TryGetLocalization(), key, word); } private static void AddWordToLocalization(Localization localization, string key, string word) { if (localization != null) { Reflections.InvokeMethod(localization, "AddWord", key, word); EvictLocalizationCache(localization); } } private static void EvictLocalizationCache(Localization localization) { try { object field = Reflections.GetField(localization, "m_cache"); if (field != null) { Reflections.InvokeMethod(field, "EvictAll"); } } catch (Exception arg) { if (!_localizationCacheWarningLogged) { Debug.LogWarning((object)$"[ModUtils] Failed to evict Localization cache after adding a word. Game-localized text may stay stale until the next language reload. {arg}"); _localizationCacheWarningLogged = true; } } } internal static string GetTranslationKey(string prefix, string internalName) { if (string.IsNullOrEmpty(internalName)) { return ""; } return internalName[0] switch { '$' => internalName.Substring(1), '@' => prefix + "_" + internalName.Substring(1), _ => internalName, }; } internal static string Translate(string prefix, string word) { return InvokeTranslate(GetTranslationKey(prefix, word)); } public static bool IsInternalName(string text) { if (!string.IsNullOrEmpty(text)) { return InternalNamePattern.IsMatch(text); } return false; } private string GetTranslationKey(string internalName) { return GetTranslationKey(_prefix, internalName); } public void AddWord(string key, string word) { InvokeAddRuntimeWord(GetTranslationKey(key), word); } internal void AddFileWord(string key, string word) { InvokeAddWord(GetTranslationKey(key), word); } public string Translate(string word) { return InvokeTranslate(GetTranslationKey(word)); } public string TranslateInternalName(string internalName) { if (IsInternalName(internalName)) { return InvokeTranslate(GetTranslationKey(internalName)); } return internalName; } public string Localize(string text) { if (string.IsNullOrEmpty(text)) { return text; } StringBuilder stringBuilder = new StringBuilder(); int num = 0; foreach (Match item in WordPattern.Matches(text)) { GroupCollection groups = item.Groups; string word = ((groups[1].Value == "@") ? (_prefix + "_" + groups[2].Value) : groups[2].Value); stringBuilder.Append(text.Substring(num, groups[0].Index - num)); stringBuilder.Append(InvokeTranslate(word)); num = groups[0].Index + groups[0].Value.Length; } stringBuilder.Append(text.Substring(num)); return stringBuilder.ToString(); } public string Localize(string text, params object[] args) { object[] array = args ?? new object[0]; return InvokeInsertWords(Localize(text), Array.ConvertAll(array, (object arg) => (arg != null) ? ((!(arg is string internalName)) ? arg.ToString() : TranslateInternalName(internalName)) : "")); } public string LocalizeTextOnly(string text, params object[] args) { object[] array = args ?? new object[0]; return InvokeInsertWords(Localize(text), Array.ConvertAll(array, delegate(object arg) { object obj; if (arg != null) { obj = arg as string; if (obj == null) { return arg.ToString(); } } else { obj = ""; } return (string)obj; })); } } public class TranslationsLoader { private static readonly string DefaultLanguage; private static readonly string JsonFilePattern; private readonly L10N _localization; private Dictionary _cache; private Logger _logger; static TranslationsLoader() { DefaultLanguage = "English"; JsonFilePattern = "*.json"; } public TranslationsLoader(L10N localization) { _localization = localization; } public void SetDebugLogger(Logger logger) { _logger = logger; } private bool LoadAllFile(string directory, string filePattern, string language, Func loading) { _logger?.Debug("Load translation files for " + language + " from directory: [directory: " + directory + ", file pattern: " + filePattern + "]"); return Directory.EnumerateFiles(directory, filePattern, SearchOption.AllDirectories).Count((string path) => loading(path, language)) > 0; } public void LoadTranslations(string languagesDir, string language) { _cache = new Dictionary(); if (!Directory.Exists(languagesDir)) { _logger?.Error("Directory does not exist: " + languagesDir); return; } if (language != DefaultLanguage && !LoadAllFile(languagesDir, JsonFilePattern, DefaultLanguage, ReadJsonFile)) { _logger?.Warning("Directory does not contain a translation file for the default language: " + languagesDir); } if (!LoadAllFile(languagesDir, JsonFilePattern, language, ReadJsonFile)) { _logger?.Warning("Directory does not contain a translation file for the " + language + ": " + languagesDir); } _cache = null; } public void LoadTranslations(string languagesDir) { string text; try { Localization instance = Localization.instance; text = ((instance != null) ? instance.GetSelectedLanguage() : null); } catch (Exception) { text = null; } LoadTranslations(languagesDir, string.IsNullOrEmpty(text) ? DefaultLanguage : text); } private bool ReadJsonFile(string path, string language) { if (!_cache.TryGetValue(path, out var value)) { try { value = Json.Parse(File.ReadAllText(path)); _cache.Add(path, value); } catch (Exception arg) { _logger?.Error($"Failed to read Json file\n{arg}"); _cache.Add(path, default(TranslationsFile)); return false; } } if (!string.Equals(value.language, language, StringComparison.OrdinalIgnoreCase)) { return false; } _logger?.Debug("Load translations: " + path); if (value.translations == null) { _logger?.Warning("Translation file does not contain translations: " + path); return true; } foreach (KeyValuePair translation in value.translations) { _localization.AddFileWord(translation.Key, translation.Value); } return true; } } [Serializable] public struct TranslationsFile { public string language; public Dictionary translations; } public class Logger { private readonly Func _isEnabled; private readonly ManualLogSource _logger; public Logger(ManualLogSource logger, Func isEnabled) { _logger = logger; _isEnabled = isEnabled; } private void Log(LogLevel level, string message) { //IL_0006: Unknown result type (might be due to invalid IL or missing references) //IL_0014: Unknown result type (might be due to invalid IL or missing references) if (_isEnabled(level)) { _logger.Log(level, (object)message); } } private void Log(LogLevel level, Func message) { //IL_0006: Unknown result type (might be due to invalid IL or missing references) //IL_0014: Unknown result type (might be due to invalid IL or missing references) if (_isEnabled(level)) { _logger.Log(level, (object)message()); } } public void Fatal(Func message) { Log((LogLevel)1, message); } public void Fatal(string message) { Log((LogLevel)1, message); } public void Error(Func message) { Log((LogLevel)2, message); } public void Error(string message) { Log((LogLevel)2, message); } public void Warning(Func message) { Log((LogLevel)4, message); } public void Warning(string message) { Log((LogLevel)4, message); } public void Info(Func message) { Log((LogLevel)16, message); } public void Info(string message) { Log((LogLevel)16, message); } public void Message(Func message) { Log((LogLevel)8, message); } public void Message(string message) { Log((LogLevel)8, message); } public void Debug(Func message) { Log((LogLevel)32, message); } public void Debug(string message) { Log((LogLevel)32, message); } } public abstract class ObjectNode : MonoBehaviour where TNode : ObjectNode where TNetwork : ObjectNetwork { protected static readonly HashSet ObjectNodes; private TNode This => (TNode)this; public TNetwork Network { get; set; } static ObjectNode() { ObjectNodes = new HashSet(); } protected virtual void Awake() { ObjectNodes.Add(This); ((MonoBehaviour)this).Invoke("NetworkConstruction", Random.Range(1f, 2f)); } protected virtual void OnDestroy() { ObjectNodes.Remove(This); Network?.RemoveNode(This); Network = null; } protected abstract TNetwork CreateNetwork(); protected abstract bool IsConnectable(TNode other); private void NetworkConstruction() { foreach (TNode item in ObjectNodes.Where(IsConnectable)) { if (Network == item.Network) { continue; } if (Network == null) { Network = item.Network; Network.AddNode(This); continue; } if (item.Network == null) { item.Network = Network; item.Network.AddNode(item); continue; } TNetwork obj = ((Network.NodeCount >= item.Network.NodeCount) ? item.Network : Network); TNetwork network = ((Network.NodeCount >= item.Network.NodeCount) ? Network : item.Network); foreach (TNode allNode in obj.GetAllNodes()) { allNode.Network.RemoveNode(allNode); allNode.Network = network; allNode.Network.AddNode(allNode); } } if (Network == null) { Network = CreateNetwork(); Network.AddNode(This); } } } public class ObjectNetwork where TNode : ObjectNode where TNetwork : ObjectNetwork { private readonly HashSet _nodes; protected Action> OnNodeChanged { get; set; } public bool IsDirty { get; private set; } public int NodeCount => _nodes.Count; protected ObjectNetwork() { _nodes = new HashSet(); } public IEnumerable EnumerateNodes() { return _nodes; } public IEnumerable GetAllNodes() { return _nodes.ToList(); } public void AddNode(TNode node) { if (_nodes.Add(node)) { IsDirty = true; } } public void RemoveNode(TNode node) { if (_nodes.Remove(node)) { IsDirty = true; } } public void Update() { if (IsDirty) { OnNodeChanged?.Invoke(GetAllNodes()); IsDirty = false; } } } public static class Objects { private static readonly ConditionalWeakTable ZNetViewCache; private static readonly ConditionalWeakTable NameCache; private const string HoverableMarker = "\u0001HOVERABLE"; static Objects() { ZNetViewCache = new ConditionalWeakTable(); NameCache = new ConditionalWeakTable(); } public static string GetPrefabName(GameObject gameObject) { return Utils.GetPrefabName(gameObject); } public static string GetName(Component component) { if (!Object.op_Implicit((Object)(object)component)) { return ""; } if (NameCache.TryGetValue(component, out var value)) { if ((object)value == "\u0001HOVERABLE") { Hoverable component2 = component.GetComponent(); string text = ((component2 != null) ? component2.GetHoverName() : null); if (string.IsNullOrEmpty(text)) { return Utils.GetPrefabName(component.gameObject); } return text; } return value; } string text2 = (from x in ((object)component).GetType().GetFields(AccessTools.all) where x.Name == "m_name" && x.FieldType == typeof(string) select x.GetValue(component) as string).FirstOrDefault(); if (!string.IsNullOrEmpty(text2)) { NameCache.Add(component, text2); return text2; } Hoverable component3 = component.GetComponent(); if (component3 != null) { HoverText val = (HoverText)(object)((component3 is HoverText) ? component3 : null); if (val != null) { text2 = val.m_text; if (!string.IsNullOrEmpty(text2)) { NameCache.Add(component, text2); return text2; } } else { text2 = component3.GetHoverName(); if (!string.IsNullOrEmpty(text2)) { NameCache.Add(component, "\u0001HOVERABLE"); return text2; } } } text2 = Utils.GetPrefabName(component.gameObject); NameCache.Add(component, text2); return text2; } public static bool GetZNetView(Component component, out ZNetView zNetView) { if (!Object.op_Implicit((Object)(object)component)) { zNetView = null; return false; } if (ZNetViewCache.TryGetValue(component, out zNetView)) { return Object.op_Implicit((Object)(object)zNetView); } zNetView = (from x in ((object)component).GetType().GetFields(AccessTools.all) where x.Name == "m_nview" && x.FieldType == typeof(ZNetView) select x).Select(delegate(FieldInfo x) { object? value = x.GetValue(component); return (ZNetView)((value is ZNetView) ? value : null); }).FirstOrDefault() ?? component.GetComponent(); if ((Object)(object)zNetView == (Object)null) { return false; } ZNetViewCache.Add(component, zNetView); return true; } public static bool GetZdoid(Component component, out ZDOID id) { //IL_0020: Unknown result type (might be due to invalid IL or missing references) //IL_0018: Unknown result type (might be due to invalid IL or missing references) //IL_0025: Unknown result type (might be due to invalid IL or missing references) //IL_002b: Unknown result type (might be due to invalid IL or missing references) //IL_0030: Unknown result type (might be due to invalid IL or missing references) id = (GetZNetView(component, out var zNetView) ? zNetView.GetZDO() : null)?.m_uid ?? ZDOID.None; return id != ZDOID.None; } public static bool HasValidOwnership(ZNetView zNetView) { if ((Object)(object)zNetView != (Object)null && zNetView.GetZDO() != null && zNetView.IsValid()) { return zNetView.IsOwner(); } return false; } public static bool HasValidOwnership(Component component) { ZNetView zNetView; return HasValidOwnership(component, out zNetView); } public static bool HasValidOwnership(Component component, out ZNetView zNetView) { if (GetZNetView(component, out zNetView)) { return HasValidOwnership(zNetView); } return false; } public static float Distance(Component obj1, Component obj2) { //IL_0006: Unknown result type (might be due to invalid IL or missing references) //IL_0011: Unknown result type (might be due to invalid IL or missing references) return Vector3.Distance(obj1.transform.position, obj2.transform.position); } public static List<(Collider collider, T obj, float distance)> GetInsideSphere(Vector3 origin, float radius, Func convertor, Collider[] buffer, int layerMask = -1) { //IL_0006: Unknown result type (might be due to invalid IL or missing references) //IL_002f: Unknown result type (might be due to invalid IL or missing references) //IL_0036: Unknown result type (might be due to invalid IL or missing references) List<(Collider, T, float)> list = new List<(Collider, T, float)>(); int num = Physics.OverlapSphereNonAlloc(origin, radius, buffer, layerMask); for (int i = 0; i < num; i++) { Collider val = buffer[i]; T val2 = convertor(val); if (val2 != null) { list.Add((val, val2, Vector3.Distance(origin, ((Component)val).transform.position))); } } return list; } public static List<(Collider collider, T obj, float distance)> GetInsideSphere(Vector3 origin, float radius, Func convertor, int bufferSize = 128, int layerMask = -1) { //IL_0000: Unknown result type (might be due to invalid IL or missing references) return GetInsideSphere(origin, radius, convertor, (Collider[])(object)new Collider[bufferSize], layerMask); } } public static class Reflections { public static TR InvokeStaticMethod(string methodName, params object[] args) { return Traverse.Create().Method(methodName, args).GetValue(args); } public static void InvokeStaticMethod(string methodName, params object[] args) { Traverse.Create().Method(methodName, args).GetValue(args); } public static TR InvokeStaticMethod(string methodName) { return Traverse.Create().Method(methodName, Array.Empty()).GetValue(); } public static void InvokeStaticMethod(string methodName) { Traverse.Create().Method(methodName, Array.Empty()).GetValue(); } public static T InvokeMethod(object instance, string methodName, params object[] args) { return Traverse.Create(instance).Method(methodName, args).GetValue(args); } public static void InvokeMethod(object instance, string methodName, params object[] args) { Traverse.Create(instance).Method(methodName, args).GetValue(args); } public static T InvokeMethod(object instance, string methodName) { return Traverse.Create(instance).Method(methodName, Array.Empty()).GetValue(); } public static void InvokeMethod(object instance, string methodName) { Traverse.Create(instance).Method(methodName, Array.Empty()).GetValue(); } public static TType GetStaticField(string fieldName) { return Traverse.Create().Field(fieldName).Value; } public static TType SetStaticField(string fieldName, TType value) { return Traverse.Create().Field(fieldName).Value = value; } public static T GetField(object instance, string fieldName) { return Traverse.Create(instance).Field(fieldName).Value; } public static void SetField(object instance, string fieldName, T value) { Traverse.Create(instance).Field(fieldName).Value = value; } } public class SpriteLoader { private static readonly Dictionary TextureCache; private Logger _logger; static SpriteLoader() { TextureCache = new Dictionary(); } private static Sprite CreateSprite(Texture2D texture, int width, int height) { //IL_000f: Unknown result type (might be due to invalid IL or missing references) //IL_0014: Unknown result type (might be due to invalid IL or missing references) return Sprite.Create(texture, new Rect(0f, 0f, (float)width, (float)height), Vector2.zero); } public static string GetTextureFileName(Sprite sprite) { using (IEnumerator> enumerator = TextureCache.Where((KeyValuePair pair) => (Object)(object)pair.Value == (Object)(object)sprite.texture).GetEnumerator()) { if (enumerator.MoveNext()) { return Path.GetFileName(enumerator.Current.Key); } } return ""; } public void SetDebugLogger(Logger logger) { _logger = logger; } public Sprite Load(string texturePath, int width, int height) { //IL_004c: Unknown result type (might be due to invalid IL or missing references) //IL_0052: Expected O, but got Unknown if (!File.Exists(texturePath)) { return null; } if (TextureCache.TryGetValue(texturePath, out var value)) { if (!((Object)(object)value != (Object)null)) { return null; } return CreateSprite(value, width, height); } try { _logger?.Info("Try to create sprite: " + texturePath); Texture2D val = new Texture2D(0, 0); ImageConversion.LoadImage(val, File.ReadAllBytes(texturePath)); TextureCache.Add(texturePath, val); return CreateSprite(val, width, height); } catch (Exception arg) { _logger?.Error($"Failed to create sprite: {texturePath}\n{arg}"); TextureCache.Add(texturePath, null); return null; } } } } namespace BetterPortal { [BepInPlugin("net.eidee.valheim.better_portal", "Better Portal", "1.0.7")] public class UnityPlugin : BaseUnityPlugin { private const string ModId = "net.eidee.valheim.better_portal"; private const string ModName = "Better Portal"; private const string ModVersion = "1.0.7"; private void Awake() { BetterPortal.Initialize(((BaseUnityPlugin)this).Info, ((BaseUnityPlugin)this).Logger, ((BaseUnityPlugin)this).Config); } } internal static class BetterPortal { public static string ModLocation { get; private set; } public static Logger Logger { get; private set; } public static L10N L10N { get; private set; } public static ConfigEntry ModifierKey { get; private set; } public static void Initialize(PluginInfo info, ManualLogSource logger, ConfigFile config) { //IL_009b: Unknown result type (might be due to invalid IL or missing references) ModLocation = Path.GetDirectoryName(info.Location) ?? ""; Logger = new Logger(logger, (LogLevel level) => (int)level != 32 && (int)level != 8); L10N = new L10N("better_portal"); L10N.AddTranslationDirectory(Path.Combine(ModLocation, "Languages")); ModifierKey = new Configuration(config, L10N).Bind("general", "ModifierKey", (KeyCode)304); ValidateModifierKey(); UpdateModifierKeyDisplay(ModifierKey.Value); ModifierKey.SettingChanged += delegate { //IL_000a: Unknown result type (might be due to invalid IL or missing references) ValidateModifierKey(); UpdateModifierKeyDisplay(ModifierKey.Value); }; Harmony.CreateAndPatchAll(Assembly.GetExecutingAssembly(), info.Metadata.GUID); } public static bool IsModifierKeyPressed() { //IL_0005: Unknown result type (might be due to invalid IL or missing references) //IL_000a: Unknown result type (might be due to invalid IL or missing references) //IL_000b: Unknown result type (might be due to invalid IL or missing references) //IL_0010: Unknown result type (might be due to invalid IL or missing references) //IL_0018: Unknown result type (might be due to invalid IL or missing references) //IL_001b: Unknown result type (might be due to invalid IL or missing references) KeyCode value = ModifierKey.Value; KeyCode pairedKey = GetPairedKey(value); if (!Input.GetKey(value)) { if ((int)pairedKey != 0) { return Input.GetKey(pairedKey); } return false; } return true; } private static void ValidateModifierKey() { //IL_0005: Unknown result type (might be due to invalid IL or missing references) //IL_001b: Unknown result type (might be due to invalid IL or missing references) if ((int)ModifierKey.Value == 0) { ModifierKey.Value = (KeyCode)((ConfigEntryBase)ModifierKey).DefaultValue; } } private static void UpdateModifierKeyDisplay(KeyCode keyCode) { //IL_000a: Unknown result type (might be due to invalid IL or missing references) L10N.AddWord("@modifier_key", GetKeyDisplayName(keyCode)); } private static KeyCode GetPairedKey(KeyCode key) { //IL_0000: Unknown result type (might be due to invalid IL or missing references) //IL_0006: Unknown result type (might be due to invalid IL or missing references) //IL_0024: Expected I4, but got Unknown return (KeyCode)((key - 303) switch { 1 => 303, 0 => 304, 3 => 305, 2 => 306, 5 => 307, 4 => 308, _ => 0, }); } private static string GetKeyDisplayName(KeyCode key) { //IL_0000: Unknown result type (might be due to invalid IL or missing references) //IL_0006: Unknown result type (might be due to invalid IL or missing references) //IL_0024: Expected I4, but got Unknown switch (key - 303) { case 0: case 1: return "Shift"; case 2: case 3: return "Ctrl"; case 4: case 5: return "Alt"; default: return ((object)(KeyCode)(ref key)).ToString(); } } } [HarmonyPatch] internal static class Patches { private static bool _transpilerApplied; private static bool _findRandomOverrideLogged; [HarmonyTranspiler] [HarmonyPatch(typeof(Game), "ConnectPortals")] private static IEnumerable Game_ConnectPortals_Transpiler(IEnumerable instructions) { List list = instructions.ToList(); List list2 = ((IEnumerable)list).Select((Func)((CodeInstruction instruction) => new CodeInstruction(instruction))).ToList(); FieldInfo sourceTagField = AccessTools.Field(typeof(ZDOVars), "s_tag"); FieldInfo operand = AccessTools.Field(typeof(ZdoTags), "DestTag"); MethodInfo getStringWithDefault = AccessTools.Method(typeof(ZDO), "GetString", new Type[2] { typeof(int), typeof(string) }, (Type[])null); MethodInfo addToCurrentlyConnectingPortals = AccessTools.Method(typeof(Game), "AddToCurrentlyConnectingPortals", (Type[])null, (Type[])null); MethodInfo setConnection = AccessTools.Method(typeof(Game), "SetConnection", new Type[3] { typeof(ZDO), typeof(ZDOID), typeof(bool) }, (Type[])null); FieldInfo uidField = AccessTools.Field(typeof(ZDO), "m_uid"); int num = 0; for (int i = 0; i <= list2.Count - 4; i++) { if (IsDestTagReadStore(list2, i, sourceTagField, getStringWithDefault)) { list2[i].operand = operand; num++; } } int num2 = FindAddToCurrentlyConnectingPortalsIndex(list2, addToCurrentlyConnectingPortals); if (num2 >= 3) { NopInstructions(list2, num2 - 3, 4); } int num3 = ((num2 >= 3) ? FindReverseSetConnectionIndex(list2, num2, setConnection, uidField) : (-1)); if (num3 >= 5) { NopInstructions(list2, num3 - 5, 6); } if (num != 2 || num2 < 3 || num3 < 5) { BetterPortal.Logger.Error("Valheim 0.221.12 compatibility patch not applied completely. " + $"DestTag replacements={num}/2, " + "AddToCurrentlyConnectingPortals=" + ((num2 >= 0) ? "ok" : "missing") + ", Reverse SetConnection=" + ((num3 >= 0) ? "ok" : "missing") + "."); return list; } _transpilerApplied = true; BetterPortal.Logger.Info("Applied Valheim 0.221.12 compatibility patch: DestTag replacements=2, AddToCurrentlyConnectingPortals disabled=1, reverse SetConnection disabled=1."); return list2; } [HarmonyPrefix] [HarmonyPatch(typeof(Game), "FindRandomUnconnectedPortal")] private static bool Game_FindRandomUnconnectedPortal_Prefix(ref ZDO __result, List portals, ZDO skip, string tag) { if (!_transpilerApplied) { return true; } if (!_findRandomOverrideLogged) { BetterPortal.Logger.Info("Applied BetterPortal FindRandomUnconnectedPortal override."); _findRandomOverrideLogged = true; } List list = portals.Where((ZDO portal) => portal != skip && portal.GetString(ZDOVars.s_tag, "") == tag).ToList(); __result = ((list.Count == 0) ? null : list[Random.Range(0, list.Count)]); return false; } [HarmonyPostfix] [HarmonyPatch(typeof(TeleportWorld), "Awake")] private static void TeleportWorld_Awake_Postfix(TeleportWorld __instance, ZNetView ___m_nview) { if (___m_nview.GetZDO() != null) { ((Component)__instance).gameObject.AddComponent(); } } [HarmonyPrefix] [HarmonyPatch(typeof(TeleportWorld), "GetHoverText")] private static bool TeleportWorld_GetHoverText_Prefix(TeleportWorld __instance, ZNetView ___m_nview, ref string __result) { //IL_0056: Unknown result type (might be due to invalid IL or missing references) //IL_005b: Unknown result type (might be due to invalid IL or missing references) ZDO zDO = ___m_nview.GetZDO(); if (zDO == null) { return true; } string text = __instance.GetText(); if (string.IsNullOrEmpty(text)) { text = BetterPortal.L10N.Translate("@empty_tag"); } string text2 = zDO.GetString(ZdoTags.DestTag, ""); if (string.IsNullOrEmpty(text2)) { text2 = BetterPortal.L10N.Translate("@empty_tag"); } ZDOID connectionZDOID = zDO.GetConnectionZDOID((ConnectionType)1); string text3 = (((ZDOID)(ref connectionZDOID)).IsNone() ? "$piece_portal_unconnected" : "$piece_portal_connected"); __result = BetterPortal.L10N.Localize("$piece_portal_tag:\"" + text + "\" @piece_portal_dest:\"" + text2 + "\" [" + text3 + "]\n[$KEY_Use] $piece_portal_settag\n[@modifier_key + $KEY_Use] @piece_portal_setdesttag"); return false; } [HarmonyPrefix] [HarmonyPatch(typeof(TeleportWorld), "Interact")] private static bool TeleportWorld_Interact_Prefix(TeleportWorld __instance, ZNetView ___m_nview, ref bool __result, Humanoid human, bool hold, bool alt) { //IL_000f: Unknown result type (might be due to invalid IL or missing references) if (hold) { __result = false; return false; } if (!PrivateArea.CheckAccess(((Component)__instance).transform.position, 0f, true, false)) { ((Character)human).Message((MessageType)2, "$piece_noaccess", 0, (Sprite)null); __result = true; return false; } if (BetterPortal.IsModifierKeyPressed()) { TextInput.instance.RequestText((TextReceiver)(object)((Component)__instance).GetComponent(), BetterPortal.L10N.Translate("@piece_portal_dest"), 10); } else { TextInput.instance.RequestText((TextReceiver)(object)__instance, "$piece_portal_tag", 10); } __result = true; return false; } [HarmonyPostfix] [HarmonyPatch(typeof(TextInput), "Update")] private static void TextInput_Update_Postfix(TextInput __instance, TextReceiver ___m_queuedSign, bool ___m_visibleFrame) { if (___m_visibleFrame && !Console.IsVisible() && !Chat.instance.HasFocus() && Object.op_Implicit((Object)(object)__instance.m_inputField) && ((TMP_InputField)__instance.m_inputField).isFocused && TeleportWorldExtension.GetAllInstance().Any((TeleportWorldExtension x) => x == ___m_queuedSign)) { TextInputExtension.Update(__instance); } } private static bool IsDestTagReadStore(IReadOnlyList instructions, int start, FieldInfo sourceTagField, MethodInfo getStringWithDefault) { if (start + 3 < instructions.Count && instructions[start].opcode == OpCodes.Ldsfld && object.Equals(instructions[start].operand, sourceTagField) && instructions[start + 1].opcode == OpCodes.Ldstr && object.Equals(instructions[start + 1].operand, "") && instructions[start + 2].opcode == OpCodes.Callvirt && object.Equals(instructions[start + 2].operand, getStringWithDefault)) { return IsStoreLocal(instructions[start + 3].opcode); } return false; } private static int FindAddToCurrentlyConnectingPortalsIndex(IReadOnlyList instructions, MethodInfo addToCurrentlyConnectingPortals) { for (int i = 3; i < instructions.Count; i++) { if (!(instructions[i].opcode != OpCodes.Call) && object.Equals(instructions[i].operand, addToCurrentlyConnectingPortals) && !(instructions[i - 3].opcode != OpCodes.Ldarg_0) && IsLoadLocal(instructions[i - 2].opcode) && IsLoadLocal(instructions[i - 1].opcode)) { return i; } } return -1; } private static int FindReverseSetConnectionIndex(IReadOnlyList instructions, int addToCurrentlyConnectingIndex, MethodInfo setConnection, FieldInfo uidField) { int num = 0; for (int i = addToCurrentlyConnectingIndex + 1; i < instructions.Count; i++) { if (i >= 5 && !(instructions[i].opcode != OpCodes.Call) && object.Equals(instructions[i].operand, setConnection) && !(instructions[i - 5].opcode != OpCodes.Ldarg_0) && IsLoadLocal(instructions[i - 4].opcode) && IsLoadLocal(instructions[i - 3].opcode) && !(instructions[i - 2].opcode != OpCodes.Ldfld) && object.Equals(instructions[i - 2].operand, uidField) && !(instructions[i - 1].opcode != OpCodes.Ldc_I4_0)) { num++; if (num == 2) { return i; } } } return -1; } private static void NopInstructions(IList instructions, int start, int length) { for (int i = start; i < start + length; i++) { instructions[i] = Nop(instructions[i]); } } private static CodeInstruction Nop(CodeInstruction instruction) { //IL_0006: Unknown result type (might be due to invalid IL or missing references) //IL_000b: Unknown result type (might be due to invalid IL or missing references) //IL_0017: Unknown result type (might be due to invalid IL or missing references) //IL_0024: Expected O, but got Unknown return new CodeInstruction(OpCodes.Nop, (object)null) { labels = instruction.labels, blocks = instruction.blocks }; } private static bool IsLoadLocal(OpCode opcode) { if (!(opcode == OpCodes.Ldloc) && !(opcode == OpCodes.Ldloc_S) && !(opcode == OpCodes.Ldloc_0) && !(opcode == OpCodes.Ldloc_1) && !(opcode == OpCodes.Ldloc_2)) { return opcode == OpCodes.Ldloc_3; } return true; } private static bool IsStoreLocal(OpCode opcode) { if (!(opcode == OpCodes.Stloc) && !(opcode == OpCodes.Stloc_S) && !(opcode == OpCodes.Stloc_0) && !(opcode == OpCodes.Stloc_1) && !(opcode == OpCodes.Stloc_2)) { return opcode == OpCodes.Stloc_3; } return true; } } [DisallowMultipleComponent] internal class TeleportWorldExtension : MonoBehaviour, TextReceiver { private static readonly List AllInstance; private ZNetView _zNetView; static TeleportWorldExtension() { AllInstance = new List(); } public static IEnumerable GetAllInstance() { return AllInstance; } private void Awake() { _zNetView = ((Component)this).GetComponent(); _zNetView.Register("SetTagDest", (Action)RPC_SetTagDest); AllInstance.Add(this); } private void OnDestroy() { AllInstance.Remove(this); _zNetView = null; } private IEnumerator UpdateConnection() { ZDO zdo = _zNetView.GetZDO(); string dest = zdo.GetString(ZdoTags.DestTag, ""); ZDOID connectionZDOID = zdo.GetConnectionZDOID((ConnectionType)1); if (!((ZDOID)(ref connectionZDOID)).IsNone()) { ZDO zDO = ZDOMan.instance.GetZDO(connectionZDOID); if (zDO == null || zDO.GetString(ZDOVars.s_tag, "") != dest) { zdo.SetOwner(ZDOMan.GetSessionID()); zdo.UpdateConnection((ConnectionType)1, ZDOID.None); ZDOMan.instance.ForceSendZDO(zdo.m_uid); } } ZDOID connectionZDOID2 = zdo.GetConnectionZDOID((ConnectionType)1); if (!((ZDOID)(ref connectionZDOID2)).IsNone()) { ((MonoBehaviour)this).StopCoroutine("UpdateConnection"); yield return null; } List list = (from x in ZDOMan.instance.GetPortals() where x != zdo && x.GetString(ZDOVars.s_tag, "") == dest select x).ToList(); ZDO val = ((list.Count == 0) ? null : list[Random.Range(0, list.Count)]); if (val != null) { zdo.SetOwner(ZDOMan.GetSessionID()); zdo.UpdateConnection((ConnectionType)1, val.m_uid); ZDOMan.instance.ForceSendZDO(zdo.m_uid); } ((MonoBehaviour)this).StopCoroutine("UpdateConnection"); yield return null; } public string GetText() { return _zNetView.GetZDO().GetString(ZdoTags.DestTag, ""); } public void SetText(string text) { if (_zNetView.IsValid()) { _zNetView.InvokeRPC("SetTagDest", new object[1] { text }); } } private void RPC_SetTagDest(long sender, string tagDest) { if (_zNetView.IsValid() && _zNetView.IsOwner() && !(GetText() == tagDest)) { _zNetView.GetZDO().Set(ZdoTags.DestTag, tagDest); ((MonoBehaviour)this).StartCoroutine("UpdateConnection"); } } } internal static class TextInputExtension { private static bool _keyPressed; private static bool _keyHold; private static KeyCode _pressedKey; private static string _originalWord; private static string _previousResult; private static List GetPortalTags() { return (from x in ZDOMan.instance.GetPortals() select x.GetString(ZDOVars.s_tag, "") into x orderby x select x).Distinct().ToList(); } private static void UpdateTextInput(TextInput input, string text) { if (Object.op_Implicit((Object)(object)input.m_inputField)) { GuiInputField inputField = input.m_inputField; if (!(text == ((TMP_InputField)inputField).text)) { ((TMP_InputField)inputField).text = (string.IsNullOrEmpty(text) ? "" : text); ((TMP_InputField)inputField).MoveTextEnd(false); } } } private static void InputUpdate() { //IL_0063: Unknown result type (might be due to invalid IL or missing references) //IL_0083: Unknown result type (might be due to invalid IL or missing references) //IL_0018: Unknown result type (might be due to invalid IL or missing references) //IL_0030: Unknown result type (might be due to invalid IL or missing references) //IL_0055: Unknown result type (might be due to invalid IL or missing references) //IL_005b: Invalid comparison between Unknown and I4 //IL_0050: Unknown result type (might be due to invalid IL or missing references) //IL_0048: Unknown result type (might be due to invalid IL or missing references) if (!_keyPressed) { if (Input.GetKeyDown((KeyCode)277)) { _pressedKey = (KeyCode)277; } else if (Input.GetKeyDown((KeyCode)273)) { _pressedKey = (KeyCode)273; } else if (Input.GetKeyDown((KeyCode)274)) { _pressedKey = (KeyCode)274; } else { _pressedKey = (KeyCode)0; } _keyPressed = (int)_pressedKey > 0; } else if (!Input.GetKeyUp(_pressedKey)) { _keyHold = true; } else { _keyPressed = false; _keyHold = false; _pressedKey = (KeyCode)0; } } public static void Update(TextInput input) { //IL_0034: Unknown result type (might be due to invalid IL or missing references) //IL_003e: Invalid comparison between Unknown and I4 //IL_004d: Unknown result type (might be due to invalid IL or missing references) //IL_0057: Invalid comparison between Unknown and I4 //IL_0067: Unknown result type (might be due to invalid IL or missing references) //IL_0071: Invalid comparison between Unknown and I4 //IL_0059: Unknown result type (might be due to invalid IL or missing references) //IL_0063: Invalid comparison between Unknown and I4 InputUpdate(); if (_keyPressed && !_keyHold) { string text = (Object.op_Implicit((Object)(object)input.m_inputField) ? ((TMP_InputField)input.m_inputField).text : ""); if ((int)_pressedKey == 277) { UpdateTextInput(input, AutoComplete(text)); } else if ((int)_pressedKey == 273 || (int)_pressedKey == 274) { UpdateTextInput(input, Rotate(text, (int)_pressedKey == 274)); } } } private static string AutoComplete(string word) { List portalTags = GetPortalTags(); if (word == _previousResult) { for (int i = portalTags.IndexOf(_previousResult) + 1; i < portalTags.Count; i++) { if (portalTags[i].StartsWith(_originalWord, StringComparison.OrdinalIgnoreCase)) { _previousResult = portalTags[i]; return _previousResult; } } _previousResult = null; return _originalWord; } _originalWord = null; _previousResult = null; using (IEnumerator enumerator = portalTags.Where((string tag) => tag.StartsWith(word, StringComparison.OrdinalIgnoreCase)).GetEnumerator()) { if (enumerator.MoveNext()) { string current = enumerator.Current; _originalWord = word; _previousResult = current; return _previousResult; } } return word; } private static string Rotate(string current, bool ascending) { List portalTags = GetPortalTags(); if (string.IsNullOrEmpty(current)) { if (!ascending) { return portalTags.LastOrDefault((string x) => !string.IsNullOrEmpty(x)); } return portalTags.FirstOrDefault((string x) => !string.IsNullOrEmpty(x)); } int num = portalTags.FindIndex((string x) => x.Equals(current, StringComparison.OrdinalIgnoreCase)); if (num == -1) { num = portalTags.FindIndex((string x) => x.StartsWith(current, StringComparison.OrdinalIgnoreCase)); } if (num == -1) { if (!ascending) { return portalTags.LastOrDefault(); } return portalTags.FirstOrDefault(); } num += (ascending ? 1 : (-1)); if (num < 0) { return portalTags.LastOrDefault(); } if (num >= portalTags.Count) { return portalTags.FirstOrDefault(); } return portalTags[num]; } } public static class ZdoTags { public static readonly int DestTag = StringExtensionMethods.GetStableHashCode("desttag"); } } [CompilerGenerated] internal sealed class <7cda94bd-7874-4e2c-8bfa-88f10cd02752> { internal static readonly long 52E5A2D9F0BC9DF8C1DC20A1983BE35AD3490120BF21F6328E54E10A37E9FE4C/* Not supported: data(22 00 2C 00 0D 00 0A 00) */; } namespace LitJson { internal enum JsonType { None, Object, Array, String, Int, Long, Double, Boolean } internal interface IJsonWrapper : IList, ICollection, IEnumerable, IOrderedDictionary, IDictionary { bool IsArray { get; } bool IsBoolean { get; } bool IsDouble { get; } bool IsInt { get; } bool IsLong { get; } bool IsObject { get; } bool IsString { get; } bool GetBoolean(); double GetDouble(); int GetInt(); JsonType GetJsonType(); long GetLong(); string GetString(); void SetBoolean(bool val); void SetDouble(double val); void SetInt(int val); void SetJsonType(JsonType type); void SetLong(long val); void SetString(string val); string ToJson(); void ToJson(JsonWriter writer); } internal class JsonData : IJsonWrapper, IList, ICollection, IEnumerable, IOrderedDictionary, IDictionary, IEquatable { private IList inst_array; private bool inst_boolean; private double inst_double; private int inst_int; private long inst_long; private IDictionary inst_object; private string inst_string; private string json; private JsonType type; private IList> object_list; public int Count => EnsureCollection().Count; public bool IsArray => type == JsonType.Array; public bool IsBoolean => type == JsonType.Boolean; public bool IsDouble => type == JsonType.Double; public bool IsInt => type == JsonType.Int; public bool IsLong => type == JsonType.Long; public bool IsObject => type == JsonType.Object; public bool IsString => type == JsonType.String; public ICollection Keys { get { EnsureDictionary(); return inst_object.Keys; } } int ICollection.Count => Count; bool ICollection.IsSynchronized => EnsureCollection().IsSynchronized; object ICollection.SyncRoot => EnsureCollection().SyncRoot; bool IDictionary.IsFixedSize => EnsureDictionary().IsFixedSize; bool IDictionary.IsReadOnly => EnsureDictionary().IsReadOnly; ICollection IDictionary.Keys { get { EnsureDictionary(); IList list = new List(); foreach (KeyValuePair item in object_list) { list.Add(item.Key); } return (ICollection)list; } } ICollection IDictionary.Values { get { EnsureDictionary(); IList list = new List(); foreach (KeyValuePair item in object_list) { list.Add(item.Value); } return (ICollection)list; } } bool IJsonWrapper.IsArray => IsArray; bool IJsonWrapper.IsBoolean => IsBoolean; bool IJsonWrapper.IsDouble => IsDouble; bool IJsonWrapper.IsInt => IsInt; bool IJsonWrapper.IsLong => IsLong; bool IJsonWrapper.IsObject => IsObject; bool IJsonWrapper.IsString => IsString; bool IList.IsFixedSize => EnsureList().IsFixedSize; bool IList.IsReadOnly => EnsureList().IsReadOnly; object IDictionary.this[object key] { get { return EnsureDictionary()[key]; } set { if (!(key is string)) { throw new ArgumentException("The key has to be a string"); } JsonData value2 = ToJsonData(value); this[(string)key] = value2; } } object IOrderedDictionary.this[int idx] { get { EnsureDictionary(); return object_list[idx].Value; } set { EnsureDictionary(); JsonData value2 = ToJsonData(value); KeyValuePair keyValuePair = object_list[idx]; inst_object[keyValuePair.Key] = value2; KeyValuePair value3 = new KeyValuePair(keyValuePair.Key, value2); object_list[idx] = value3; } } object IList.this[int index] { get { return EnsureList()[index]; } set { EnsureList(); JsonData value2 = ToJsonData(value); this[index] = value2; } } public JsonData this[string prop_name] { get { EnsureDictionary(); return inst_object[prop_name]; } set { EnsureDictionary(); KeyValuePair keyValuePair = new KeyValuePair(prop_name, value); if (inst_object.ContainsKey(prop_name)) { for (int i = 0; i < object_list.Count; i++) { if (object_list[i].Key == prop_name) { object_list[i] = keyValuePair; break; } } } else { object_list.Add(keyValuePair); } inst_object[prop_name] = value; json = null; } } public JsonData this[int index] { get { EnsureCollection(); if (type == JsonType.Array) { return inst_array[index]; } return object_list[index].Value; } set { EnsureCollection(); if (type == JsonType.Array) { inst_array[index] = value; } else { KeyValuePair keyValuePair = object_list[index]; KeyValuePair value2 = new KeyValuePair(keyValuePair.Key, value); object_list[index] = value2; inst_object[keyValuePair.Key] = value; } json = null; } } public bool ContainsKey(string key) { EnsureDictionary(); return inst_object.Keys.Contains(key); } public JsonData() { } public JsonData(bool boolean) { type = JsonType.Boolean; inst_boolean = boolean; } public JsonData(double number) { type = JsonType.Double; inst_double = number; } public JsonData(int number) { type = JsonType.Int; inst_int = number; } public JsonData(long number) { type = JsonType.Long; inst_long = number; } public JsonData(object obj) { if (obj is bool) { type = JsonType.Boolean; inst_boolean = (bool)obj; return; } if (obj is double) { type = JsonType.Double; inst_double = (double)obj; return; } if (obj is int) { type = JsonType.Int; inst_int = (int)obj; return; } if (obj is long) { type = JsonType.Long; inst_long = (long)obj; return; } if (obj is string) { type = JsonType.String; inst_string = (string)obj; return; } throw new ArgumentException("Unable to wrap the given object with JsonData"); } public JsonData(string str) { type = JsonType.String; inst_string = str; } public static implicit operator JsonData(bool data) { return new JsonData(data); } public static implicit operator JsonData(double data) { return new JsonData(data); } public static implicit operator JsonData(int data) { return new JsonData(data); } public static implicit operator JsonData(long data) { return new JsonData(data); } public static implicit operator JsonData(string data) { return new JsonData(data); } public static explicit operator bool(JsonData data) { if (data.type != JsonType.Boolean) { throw new InvalidCastException("Instance of JsonData doesn't hold a double"); } return data.inst_boolean; } public static explicit operator double(JsonData data) { if (data.type != JsonType.Double) { throw new InvalidCastException("Instance of JsonData doesn't hold a double"); } return data.inst_double; } public static explicit operator int(JsonData data) { if (data.type != JsonType.Int && data.type != JsonType.Long) { throw new InvalidCastException("Instance of JsonData doesn't hold an int"); } if (data.type != JsonType.Int) { return (int)data.inst_long; } return data.inst_int; } public static explicit operator long(JsonData data) { if (data.type != JsonType.Long && data.type != JsonType.Int) { throw new InvalidCastException("Instance of JsonData doesn't hold a long"); } if (data.type != JsonType.Long) { return data.inst_int; } return data.inst_long; } public static explicit operator string(JsonData data) { if (data.type != JsonType.String) { throw new InvalidCastException("Instance of JsonData doesn't hold a string"); } return data.inst_string; } void ICollection.CopyTo(Array array, int index) { EnsureCollection().CopyTo(array, index); } void IDictionary.Add(object key, object value) { JsonData value2 = ToJsonData(value); EnsureDictionary().Add(key, value2); KeyValuePair item = new KeyValuePair((string)key, value2); object_list.Add(item); json = null; } void IDictionary.Clear() { EnsureDictionary().Clear(); object_list.Clear(); json = null; } bool IDictionary.Contains(object key) { return EnsureDictionary().Contains(key); } IDictionaryEnumerator IDictionary.GetEnumerator() { return ((IOrderedDictionary)this).GetEnumerator(); } void IDictionary.Remove(object key) { EnsureDictionary().Remove(key); for (int i = 0; i < object_list.Count; i++) { if (object_list[i].Key == (string)key) { object_list.RemoveAt(i); break; } } json = null; } IEnumerator IEnumerable.GetEnumerator() { return EnsureCollection().GetEnumerator(); } bool IJsonWrapper.GetBoolean() { if (type != JsonType.Boolean) { throw new InvalidOperationException("JsonData instance doesn't hold a boolean"); } return inst_boolean; } double IJsonWrapper.GetDouble() { if (type != JsonType.Double) { throw new InvalidOperationException("JsonData instance doesn't hold a double"); } return inst_double; } int IJsonWrapper.GetInt() { if (type != JsonType.Int) { throw new InvalidOperationException("JsonData instance doesn't hold an int"); } return inst_int; } long IJsonWrapper.GetLong() { if (type != JsonType.Long) { throw new InvalidOperationException("JsonData instance doesn't hold a long"); } return inst_long; } string IJsonWrapper.GetString() { if (type != JsonType.String) { throw new InvalidOperationException("JsonData instance doesn't hold a string"); } return inst_string; } void IJsonWrapper.SetBoolean(bool val) { type = JsonType.Boolean; inst_boolean = val; json = null; } void IJsonWrapper.SetDouble(double val) { type = JsonType.Double; inst_double = val; json = null; } void IJsonWrapper.SetInt(int val) { type = JsonType.Int; inst_int = val; json = null; } void IJsonWrapper.SetLong(long val) { type = JsonType.Long; inst_long = val; json = null; } void IJsonWrapper.SetString(string val) { type = JsonType.String; inst_string = val; json = null; } string IJsonWrapper.ToJson() { return ToJson(); } void IJsonWrapper.ToJson(JsonWriter writer) { ToJson(writer); } int IList.Add(object value) { return Add(value); } void IList.Clear() { EnsureList().Clear(); json = null; } bool IList.Contains(object value) { return EnsureList().Contains(value); } int IList.IndexOf(object value) { return EnsureList().IndexOf(value); } void IList.Insert(int index, object value) { EnsureList().Insert(index, value); json = null; } void IList.Remove(object value) { EnsureList().Remove(value); json = null; } void IList.RemoveAt(int index) { EnsureList().RemoveAt(index); json = null; } IDictionaryEnumerator IOrderedDictionary.GetEnumerator() { EnsureDictionary(); return new OrderedDictionaryEnumerator(object_list.GetEnumerator()); } void IOrderedDictionary.Insert(int idx, object key, object value) { string text = (string)key; JsonData value2 = (this[text] = ToJsonData(value)); KeyValuePair item = new KeyValuePair(text, value2); object_list.Insert(idx, item); } void IOrderedDictionary.RemoveAt(int idx) { EnsureDictionary(); inst_object.Remove(object_list[idx].Key); object_list.RemoveAt(idx); } private ICollection EnsureCollection() { if (type == JsonType.Array) { return (ICollection)inst_array; } if (type == JsonType.Object) { return (ICollection)inst_object; } throw new InvalidOperationException("The JsonData instance has to be initialized first"); } private IDictionary EnsureDictionary() { if (type == JsonType.Object) { return (IDictionary)inst_object; } if (type != 0) { throw new InvalidOperationException("Instance of JsonData is not a dictionary"); } type = JsonType.Object; inst_object = new Dictionary(); object_list = new List>(); return (IDictionary)inst_object; } private IList EnsureList() { if (type == JsonType.Array) { return (IList)inst_array; } if (type != 0) { throw new InvalidOperationException("Instance of JsonData is not a list"); } type = JsonType.Array; inst_array = new List(); return (IList)inst_array; } private JsonData ToJsonData(object obj) { if (obj == null) { return null; } if (obj is JsonData) { return (JsonData)obj; } return new JsonData(obj); } private static void WriteJson(IJsonWrapper obj, JsonWriter writer) { if (obj == null) { writer.Write(null); } else if (obj.IsString) { writer.Write(obj.GetString()); } else if (obj.IsBoolean) { writer.Write(obj.GetBoolean()); } else if (obj.IsDouble) { writer.Write(obj.GetDouble()); } else if (obj.IsInt) { writer.Write(obj.GetInt()); } else if (obj.IsLong) { writer.Write(obj.GetLong()); } else if (obj.IsArray) { writer.WriteArrayStart(); foreach (JsonData item in (IEnumerable)obj) { WriteJson(item, writer); } writer.WriteArrayEnd(); } else { if (!obj.IsObject) { return; } writer.WriteObjectStart(); foreach (DictionaryEntry item2 in (IDictionary)obj) { writer.WritePropertyName((string)item2.Key); WriteJson((JsonData)item2.Value, writer); } writer.WriteObjectEnd(); } } public int Add(object value) { JsonData value2 = ToJsonData(value); json = null; return EnsureList().Add(value2); } public bool Remove(object obj) { json = null; if (IsObject) { JsonData value = null; if (inst_object.TryGetValue((string)obj, out value)) { if (inst_object.Remove((string)obj)) { return object_list.Remove(new KeyValuePair((string)obj, value)); } return false; } throw new KeyNotFoundException("The specified key was not found in the JsonData object."); } if (IsArray) { return inst_array.Remove(ToJsonData(obj)); } throw new InvalidOperationException("Instance of JsonData is not an object or a list."); } public void Clear() { if (IsObject) { ((IDictionary)this).Clear(); } else if (IsArray) { ((IList)this).Clear(); } } public bool Equals(JsonData x) { if (x == null) { return false; } if (x.type != type && ((x.type != JsonType.Int && x.type != JsonType.Long) || (type != JsonType.Int && type != JsonType.Long))) { return false; } switch (type) { case JsonType.None: return true; case JsonType.Object: return inst_object.Equals(x.inst_object); case JsonType.Array: return inst_array.Equals(x.inst_array); case JsonType.String: return inst_string.Equals(x.inst_string); case JsonType.Int: if (x.IsLong) { if (x.inst_long < int.MinValue || x.inst_long > int.MaxValue) { return false; } return inst_int.Equals((int)x.inst_long); } return inst_int.Equals(x.inst_int); case JsonType.Long: if (x.IsInt) { if (inst_long < int.MinValue || inst_long > int.MaxValue) { return false; } return x.inst_int.Equals((int)inst_long); } return inst_long.Equals(x.inst_long); case JsonType.Double: return inst_double.Equals(x.inst_double); case JsonType.Boolean: return inst_boolean.Equals(x.inst_boolean); default: return false; } } public JsonType GetJsonType() { return type; } public void SetJsonType(JsonType type) { if (this.type != type) { switch (type) { case JsonType.Object: inst_object = new Dictionary(); object_list = new List>(); break; case JsonType.Array: inst_array = new List(); break; case JsonType.String: inst_string = null; break; case JsonType.Int: inst_int = 0; break; case JsonType.Long: inst_long = 0L; break; case JsonType.Double: inst_double = 0.0; break; case JsonType.Boolean: inst_boolean = false; break; } this.type = type; } } public string ToJson() { if (json != null) { return json; } StringWriter stringWriter = new StringWriter(); JsonWriter jsonWriter = new JsonWriter(stringWriter); jsonWriter.Validate = false; WriteJson(this, jsonWriter); json = stringWriter.ToString(); return json; } public void ToJson(JsonWriter writer) { bool validate = writer.Validate; writer.Validate = false; WriteJson(this, writer); writer.Validate = validate; } public override string ToString() { return type switch { JsonType.Array => "JsonData array", JsonType.Boolean => inst_boolean.ToString(), JsonType.Double => inst_double.ToString(), JsonType.Int => inst_int.ToString(), JsonType.Long => inst_long.ToString(), JsonType.Object => "JsonData object", JsonType.String => inst_string, _ => "Uninitialized JsonData", }; } } internal class OrderedDictionaryEnumerator : IDictionaryEnumerator, IEnumerator { private IEnumerator> list_enumerator; public object Current => Entry; public DictionaryEntry Entry { get { KeyValuePair current = list_enumerator.Current; return new DictionaryEntry(current.Key, current.Value); } } public object Key => list_enumerator.Current.Key; public object Value => list_enumerator.Current.Value; public OrderedDictionaryEnumerator(IEnumerator> enumerator) { list_enumerator = enumerator; } public bool MoveNext() { return list_enumerator.MoveNext(); } public void Reset() { list_enumerator.Reset(); } } internal class JsonException : ApplicationException { public JsonException() { } internal JsonException(ParserToken token) : base($"Invalid token '{token}' in input string") { } internal JsonException(ParserToken token, Exception inner_exception) : base($"Invalid token '{token}' in input string", inner_exception) { } internal JsonException(int c) : base($"Invalid character '{(char)c}' in input string") { } internal JsonException(int c, Exception inner_exception) : base($"Invalid character '{(char)c}' in input string", inner_exception) { } public JsonException(string message) : base(message) { } public JsonException(string message, Exception inner_exception) : base(message, inner_exception) { } } internal struct PropertyMetadata { public MemberInfo Info; public bool IsField; public Type Type; } internal struct ArrayMetadata { private Type element_type; private bool is_array; private bool is_list; public Type ElementType { get { if (element_type == null) { return typeof(JsonData); } return element_type; } set { element_type = value; } } public bool IsArray { get { return is_array; } set { is_array = value; } } public bool IsList { get { return is_list; } set { is_list = value; } } } internal struct ObjectMetadata { private Type element_type; private bool is_dictionary; private IDictionary properties; public Type ElementType { get { if (element_type == null) { return typeof(JsonData); } return element_type; } set { element_type = value; } } public bool IsDictionary { get { return is_dictionary; } set { is_dictionary = value; } } public IDictionary Properties { get { return properties; } set { properties = value; } } } internal delegate void ExporterFunc(object obj, JsonWriter writer); public delegate void ExporterFunc(T obj, JsonWriter writer); internal delegate object ImporterFunc(object input); public delegate TValue ImporterFunc(TJson input); internal delegate IJsonWrapper WrapperFactory(); internal class JsonMapper { private static readonly int max_nesting_depth; private static readonly IFormatProvider datetime_format; private static readonly IDictionary base_exporters_table; private static readonly IDictionary custom_exporters_table; private static readonly IDictionary> base_importers_table; private static readonly IDictionary> custom_importers_table; private static readonly IDictionary array_metadata; private static readonly object array_metadata_lock; private static readonly IDictionary> conv_ops; private static readonly object conv_ops_lock; private static readonly IDictionary object_metadata; private static readonly object object_metadata_lock; private static readonly IDictionary> type_properties; private static readonly object type_properties_lock; private static readonly JsonWriter static_writer; private static readonly object static_writer_lock; static JsonMapper() { array_metadata_lock = new object(); conv_ops_lock = new object(); object_metadata_lock = new object(); type_properties_lock = new object(); static_writer_lock = new object(); max_nesting_depth = 100; array_metadata = new Dictionary(); conv_ops = new Dictionary>(); object_metadata = new Dictionary(); type_properties = new Dictionary>(); static_writer = new JsonWriter(); datetime_format = DateTimeFormatInfo.InvariantInfo; base_exporters_table = new Dictionary(); custom_exporters_table = new Dictionary(); base_importers_table = new Dictionary>(); custom_importers_table = new Dictionary>(); RegisterBaseExporters(); RegisterBaseImporters(); } private static void AddArrayMetadata(Type type) { if (array_metadata.ContainsKey(type)) { return; } ArrayMetadata value = default(ArrayMetadata); value.IsArray = type.IsArray; if (type.GetInterface("System.Collections.IList") != null) { value.IsList = true; } PropertyInfo[] properties = type.GetProperties(); foreach (PropertyInfo propertyInfo in properties) { if (!(propertyInfo.Name != "Item")) { ParameterInfo[] indexParameters = propertyInfo.GetIndexParameters(); if (indexParameters.Length == 1 && indexParameters[0].ParameterType == typeof(int)) { value.ElementType = propertyInfo.PropertyType; } } } lock (array_metadata_lock) { try { array_metadata.Add(type, value); } catch (ArgumentException) { } } } private static void AddObjectMetadata(Type type) { if (object_metadata.ContainsKey(type)) { return; } ObjectMetadata value = default(ObjectMetadata); if (type.GetInterface("System.Collections.IDictionary") != null) { value.IsDictionary = true; } value.Properties = new Dictionary(); PropertyInfo[] properties = type.GetProperties(); foreach (PropertyInfo propertyInfo in properties) { if (propertyInfo.Name == "Item") { ParameterInfo[] indexParameters = propertyInfo.GetIndexParameters(); if (indexParameters.Length == 1 && indexParameters[0].ParameterType == typeof(string)) { value.ElementType = propertyInfo.PropertyType; } } else { PropertyMetadata value2 = default(PropertyMetadata); value2.Info = propertyInfo; value2.Type = propertyInfo.PropertyType; value.Properties.Add(propertyInfo.Name, value2); } } FieldInfo[] fields = type.GetFields(); foreach (FieldInfo fieldInfo in fields) { PropertyMetadata value3 = default(PropertyMetadata); value3.Info = fieldInfo; value3.IsField = true; value3.Type = fieldInfo.FieldType; value.Properties.Add(fieldInfo.Name, value3); } lock (object_metadata_lock) { try { object_metadata.Add(type, value); } catch (ArgumentException) { } } } private static void AddTypeProperties(Type type) { if (type_properties.ContainsKey(type)) { return; } IList list = new List(); PropertyInfo[] properties = type.GetProperties(); foreach (PropertyInfo propertyInfo in properties) { if (!(propertyInfo.Name == "Item")) { PropertyMetadata item = default(PropertyMetadata); item.Info = propertyInfo; item.IsField = false; list.Add(item); } } FieldInfo[] fields = type.GetFields(); foreach (FieldInfo info in fields) { PropertyMetadata item2 = default(PropertyMetadata); item2.Info = info; item2.IsField = true; list.Add(item2); } lock (type_properties_lock) { try { type_properties.Add(type, list); } catch (ArgumentException) { } } } private static MethodInfo GetConvOp(Type t1, Type t2) { lock (conv_ops_lock) { if (!conv_ops.ContainsKey(t1)) { conv_ops.Add(t1, new Dictionary()); } } if (conv_ops[t1].ContainsKey(t2)) { return conv_ops[t1][t2]; } MethodInfo method = t1.GetMethod("op_Implicit", new Type[1] { t2 }); lock (conv_ops_lock) { try { conv_ops[t1].Add(t2, method); return method; } catch (ArgumentException) { return conv_ops[t1][t2]; } } } private static object ReadValue(Type inst_type, JsonReader reader) { reader.Read(); if (reader.Token == JsonToken.ArrayEnd) { return null; } Type underlyingType = Nullable.GetUnderlyingType(inst_type); Type type = underlyingType ?? inst_type; if (reader.Token == JsonToken.Null) { if (inst_type.IsClass || underlyingType != null) { return null; } throw new JsonException($"Can't assign null to an instance of type {inst_type}"); } if (reader.Token == JsonToken.Double || reader.Token == JsonToken.Int || reader.Token == JsonToken.Long || reader.Token == JsonToken.String || reader.Token == JsonToken.Boolean) { Type type2 = reader.Value.GetType(); if (type.IsAssignableFrom(type2)) { return reader.Value; } if (custom_importers_table.ContainsKey(type2) && custom_importers_table[type2].ContainsKey(type)) { return custom_importers_table[type2][type](reader.Value); } if (base_importers_table.ContainsKey(type2) && base_importers_table[type2].ContainsKey(type)) { return base_importers_table[type2][type](reader.Value); } if (type.IsEnum) { return Enum.ToObject(type, reader.Value); } MethodInfo convOp = GetConvOp(type, type2); if (convOp != null) { return convOp.Invoke(null, new object[1] { reader.Value }); } throw new JsonException($"Can't assign value '{reader.Value}' (type {type2}) to type {inst_type}"); } object obj = null; if (reader.Token == JsonToken.ArrayStart) { AddArrayMetadata(inst_type); ArrayMetadata arrayMetadata = array_metadata[inst_type]; if (!arrayMetadata.IsArray && !arrayMetadata.IsList) { throw new JsonException($"Type {inst_type} can't act as an array"); } IList list; Type elementType; if (!arrayMetadata.IsArray) { list = (IList)Activator.CreateInstance(inst_type); elementType = arrayMetadata.ElementType; } else { list = new ArrayList(); elementType = inst_type.GetElementType(); } list.Clear(); while (true) { object obj2 = ReadValue(elementType, reader); if (obj2 == null && reader.Token == JsonToken.ArrayEnd) { break; } list.Add(obj2); } if (arrayMetadata.IsArray) { int count = list.Count; obj = Array.CreateInstance(elementType, count); for (int i = 0; i < count; i++) { ((Array)obj).SetValue(list[i], i); } } else { obj = list; } } else if (reader.Token == JsonToken.ObjectStart) { AddObjectMetadata(type); ObjectMetadata objectMetadata = object_metadata[type]; obj = Activator.CreateInstance(type); while (true) { reader.Read(); if (reader.Token == JsonToken.ObjectEnd) { break; } string text = (string)reader.Value; if (objectMetadata.Properties.ContainsKey(text)) { PropertyMetadata propertyMetadata = objectMetadata.Properties[text]; if (propertyMetadata.IsField) { ((FieldInfo)propertyMetadata.Info).SetValue(obj, ReadValue(propertyMetadata.Type, reader)); continue; } PropertyInfo propertyInfo = (PropertyInfo)propertyMetadata.Info; if (propertyInfo.CanWrite) { propertyInfo.SetValue(obj, ReadValue(propertyMetadata.Type, reader), null); } else { ReadValue(propertyMetadata.Type, reader); } } else if (!objectMetadata.IsDictionary) { if (!reader.SkipNonMembers) { throw new JsonException($"The type {inst_type} doesn't have the property '{text}'"); } ReadSkip(reader); } else { ((IDictionary)obj).Add(text, ReadValue(objectMetadata.ElementType, reader)); } } } return obj; } private static IJsonWrapper ReadValue(WrapperFactory factory, JsonReader reader) { reader.Read(); if (reader.Token == JsonToken.ArrayEnd || reader.Token == JsonToken.Null) { return null; } IJsonWrapper jsonWrapper = factory(); if (reader.Token == JsonToken.String) { jsonWrapper.SetString((string)reader.Value); return jsonWrapper; } if (reader.Token == JsonToken.Double) { jsonWrapper.SetDouble((double)reader.Value); return jsonWrapper; } if (reader.Token == JsonToken.Int) { jsonWrapper.SetInt((int)reader.Value); return jsonWrapper; } if (reader.Token == JsonToken.Long) { jsonWrapper.SetLong((long)reader.Value); return jsonWrapper; } if (reader.Token == JsonToken.Boolean) { jsonWrapper.SetBoolean((bool)reader.Value); return jsonWrapper; } if (reader.Token == JsonToken.ArrayStart) { jsonWrapper.SetJsonType(JsonType.Array); while (true) { IJsonWrapper jsonWrapper2 = ReadValue(factory, reader); if (jsonWrapper2 == null && reader.Token == JsonToken.ArrayEnd) { break; } jsonWrapper.Add(jsonWrapper2); } } else if (reader.Token == JsonToken.ObjectStart) { jsonWrapper.SetJsonType(JsonType.Object); while (true) { reader.Read(); if (reader.Token == JsonToken.ObjectEnd) { break; } string key = (string)reader.Value; jsonWrapper[key] = ReadValue(factory, reader); } } return jsonWrapper; } private static void ReadSkip(JsonReader reader) { ToWrapper(() => new JsonMockWrapper(), reader); } private static void RegisterBaseExporters() { base_exporters_table[typeof(byte)] = delegate(object obj, JsonWriter writer) { writer.Write(Convert.ToInt32((byte)obj)); }; base_exporters_table[typeof(char)] = delegate(object obj, JsonWriter writer) { writer.Write(Convert.ToString((char)obj)); }; base_exporters_table[typeof(DateTime)] = delegate(object obj, JsonWriter writer) { writer.Write(Convert.ToString((DateTime)obj, datetime_format)); }; base_exporters_table[typeof(decimal)] = delegate(object obj, JsonWriter writer) { writer.Write((decimal)obj); }; base_exporters_table[typeof(sbyte)] = delegate(object obj, JsonWriter writer) { writer.Write(Convert.ToInt32((sbyte)obj)); }; base_exporters_table[typeof(short)] = delegate(object obj, JsonWriter writer) { writer.Write(Convert.ToInt32((short)obj)); }; base_exporters_table[typeof(ushort)] = delegate(object obj, JsonWriter writer) { writer.Write(Convert.ToInt32((ushort)obj)); }; base_exporters_table[typeof(uint)] = delegate(object obj, JsonWriter writer) { writer.Write(Convert.ToUInt64((uint)obj)); }; base_exporters_table[typeof(ulong)] = delegate(object obj, JsonWriter writer) { writer.Write((ulong)obj); }; base_exporters_table[typeof(DateTimeOffset)] = delegate(object obj, JsonWriter writer) { writer.Write(((DateTimeOffset)obj).ToString("yyyy-MM-ddTHH:mm:ss.fffffffzzz", datetime_format)); }; } private static void RegisterBaseImporters() { ImporterFunc importer = (object input) => Convert.ToByte((int)input); RegisterImporter(base_importers_table, typeof(int), typeof(byte), importer); importer = (object input) => Convert.ToUInt64((int)input); RegisterImporter(base_importers_table, typeof(int), typeof(ulong), importer); importer = (object input) => Convert.ToInt64((int)input); RegisterImporter(base_importers_table, typeof(int), typeof(long), importer); importer = (object input) => Convert.ToSByte((int)input); RegisterImporter(base_importers_table, typeof(int), typeof(sbyte), importer); importer = (object input) => Convert.ToInt16((int)input); RegisterImporter(base_importers_table, typeof(int), typeof(short), importer); importer = (object input) => Convert.ToUInt16((int)input); RegisterImporter(base_importers_table, typeof(int), typeof(ushort), importer); importer = (object input) => Convert.ToUInt32((int)input); RegisterImporter(base_importers_table, typeof(int), typeof(uint), importer); importer = (object input) => Convert.ToSingle((int)input); RegisterImporter(base_importers_table, typeof(int), typeof(float), importer); importer = (object input) => Convert.ToDouble((int)input); RegisterImporter(base_importers_table, typeof(int), typeof(double), importer); importer = (object input) => Convert.ToDecimal((double)input); RegisterImporter(base_importers_table, typeof(double), typeof(decimal), importer); importer = (object input) => Convert.ToSingle((double)input); RegisterImporter(base_importers_table, typeof(double), typeof(float), importer); importer = (object input) => Convert.ToUInt32((long)input); RegisterImporter(base_importers_table, typeof(long), typeof(uint), importer); importer = (object input) => Convert.ToChar((string)input); RegisterImporter(base_importers_table, typeof(string), typeof(char), importer); importer = (object input) => Convert.ToDateTime((string)input, datetime_format); RegisterImporter(base_importers_table, typeof(string), typeof(DateTime), importer); importer = (object input) => DateTimeOffset.Parse((string)input, datetime_format); RegisterImporter(base_importers_table, typeof(string), typeof(DateTimeOffset), importer); } private static void RegisterImporter(IDictionary> table, Type json_type, Type value_type, ImporterFunc importer) { if (!table.ContainsKey(json_type)) { table.Add(json_type, new Dictionary()); } table[json_type][value_type] = importer; } private static void WriteValue(object obj, JsonWriter writer, bool writer_is_private, int depth) { if (depth > max_nesting_depth) { throw new JsonException($"Max allowed object depth reached while trying to export from type {obj.GetType()}"); } if (obj == null) { writer.Write(null); return; } if (obj is IJsonWrapper) { if (writer_is_private) { writer.TextWriter.Write(((IJsonWrapper)obj).ToJson()); } else { ((IJsonWrapper)obj).ToJson(writer); } return; } if (obj is string) { writer.Write((string)obj); return; } if (obj is double) { writer.Write((double)obj); return; } if (obj is float) { writer.Write((float)obj); return; } if (obj is int) { writer.Write((int)obj); return; } if (obj is bool) { writer.Write((bool)obj); return; } if (obj is long) { writer.Write((long)obj); return; } if (obj is Array) { writer.WriteArrayStart(); foreach (object item in (Array)obj) { WriteValue(item, writer, writer_is_private, depth + 1); } writer.WriteArrayEnd(); return; } if (obj is IList) { writer.WriteArrayStart(); foreach (object item2 in (IList)obj) { WriteValue(item2, writer, writer_is_private, depth + 1); } writer.WriteArrayEnd(); return; } if (obj is IDictionary dictionary) { writer.WriteObjectStart(); foreach (DictionaryEntry item3 in dictionary) { string property_name = ((item3.Key is string text) ? text : Convert.ToString(item3.Key, CultureInfo.InvariantCulture)); writer.WritePropertyName(property_name); WriteValue(item3.Value, writer, writer_is_private, depth + 1); } writer.WriteObjectEnd(); return; } Type type = obj.GetType(); if (custom_exporters_table.ContainsKey(type)) { custom_exporters_table[type](obj, writer); return; } if (base_exporters_table.ContainsKey(type)) { base_exporters_table[type](obj, writer); return; } if (obj is Enum) { Type underlyingType = Enum.GetUnderlyingType(type); if (underlyingType == typeof(long)) { writer.Write((long)obj); } else if (underlyingType == typeof(uint)) { writer.Write((uint)obj); } else if (underlyingType == typeof(ulong)) { writer.Write((ulong)obj); } else if (underlyingType == typeof(ushort)) { writer.Write((ushort)obj); } else if (underlyingType == typeof(short)) { writer.Write((short)obj); } else if (underlyingType == typeof(byte)) { writer.Write((byte)obj); } else if (underlyingType == typeof(sbyte)) { writer.Write((sbyte)obj); } else { writer.Write((int)obj); } return; } AddTypeProperties(type); IList list = type_properties[type]; writer.WriteObjectStart(); foreach (PropertyMetadata item4 in list) { if (item4.IsField) { writer.WritePropertyName(item4.Info.Name); WriteValue(((FieldInfo)item4.Info).GetValue(obj), writer, writer_is_private, depth + 1); continue; } PropertyInfo propertyInfo = (PropertyInfo)item4.Info; if (propertyInfo.CanRead) { writer.WritePropertyName(item4.Info.Name); WriteValue(propertyInfo.GetValue(obj, null), writer, writer_is_private, depth + 1); } } writer.WriteObjectEnd(); } public static string ToJson(object obj) { lock (static_writer_lock) { static_writer.Reset(); WriteValue(obj, static_writer, writer_is_private: true, 0); return static_writer.ToString(); } } public static void ToJson(object obj, JsonWriter writer) { WriteValue(obj, writer, writer_is_private: false, 0); } public static JsonData ToObject(JsonReader reader) { return (JsonData)ToWrapper(() => new JsonData(), reader); } public static JsonData ToObject(TextReader reader) { JsonReader reader2 = new JsonReader(reader); return (JsonData)ToWrapper(() => new JsonData(), reader2); } public static JsonData ToObject(string json) { return (JsonData)ToWrapper(() => new JsonData(), json); } public static T ToObject(JsonReader reader) { return (T)ReadValue(typeof(T), reader); } public static T ToObject(TextReader reader) { JsonReader reader2 = new JsonReader(reader); return (T)ReadValue(typeof(T), reader2); } public static T ToObject(string json) { JsonReader reader = new JsonReader(json); return (T)ReadValue(typeof(T), reader); } public static object ToObject(string json, Type ConvertType) { JsonReader reader = new JsonReader(json); return ReadValue(ConvertType, reader); } public static IJsonWrapper ToWrapper(WrapperFactory factory, JsonReader reader) { return ReadValue(factory, reader); } public static IJsonWrapper ToWrapper(WrapperFactory factory, string json) { JsonReader reader = new JsonReader(json); return ReadValue(factory, reader); } public static void RegisterExporter(ExporterFunc exporter) { ExporterFunc value = delegate(object obj, JsonWriter writer) { exporter((T)obj, writer); }; custom_exporters_table[typeof(T)] = value; } public static void RegisterImporter(ImporterFunc importer) { ImporterFunc importer2 = (object input) => importer((TJson)input); RegisterImporter(custom_importers_table, typeof(TJson), typeof(TValue), importer2); } public static void UnregisterExporters() { custom_exporters_table.Clear(); } public static void UnregisterImporters() { custom_importers_table.Clear(); } } internal class JsonMockWrapper : IJsonWrapper, IList, ICollection, IEnumerable, IOrderedDictionary, IDictionary { public bool IsArray => false; public bool IsBoolean => false; public bool IsDouble => false; public bool IsInt => false; public bool IsLong => false; public bool IsObject => false; public bool IsString => false; bool IList.IsFixedSize => true; bool IList.IsReadOnly => true; object IList.this[int index] { get { return null; } set { } } int ICollection.Count => 0; bool ICollection.IsSynchronized => false; object ICollection.SyncRoot => null; bool IDictionary.IsFixedSize => true; bool IDictionary.IsReadOnly => true; ICollection IDictionary.Keys => null; ICollection IDictionary.Values => null; object IDictionary.this[object key] { get { return null; } set { } } object IOrderedDictionary.this[int idx] { get { return null; } set { } } public bool GetBoolean() { return false; } public double GetDouble() { return 0.0; } public int GetInt() { return 0; } public JsonType GetJsonType() { return JsonType.None; } public long GetLong() { return 0L; } public string GetString() { return ""; } public void SetBoolean(bool val) { } public void SetDouble(double val) { } public void SetInt(int val) { } public void SetJsonType(JsonType type) { } public void SetLong(long val) { } public void SetString(string val) { } public string ToJson() { return ""; } public void ToJson(JsonWriter writer) { } int IList.Add(object value) { return 0; } void IList.Clear() { } bool IList.Contains(object value) { return false; } int IList.IndexOf(object value) { return -1; } void IList.Insert(int i, object v) { } void IList.Remove(object value) { } void IList.RemoveAt(int index) { } void ICollection.CopyTo(Array array, int index) { } IEnumerator IEnumerable.GetEnumerator() { return null; } void IDictionary.Add(object k, object v) { } void IDictionary.Clear() { } bool IDictionary.Contains(object key) { return false; } void IDictionary.Remove(object key) { } IDictionaryEnumerator IDictionary.GetEnumerator() { return null; } IDictionaryEnumerator IOrderedDictionary.GetEnumerator() { return null; } void IOrderedDictionary.Insert(int i, object k, object v) { } void IOrderedDictionary.RemoveAt(int i) { } } internal enum JsonToken { None, ObjectStart, PropertyName, ObjectEnd, ArrayStart, ArrayEnd, Int, Long, Double, String, Boolean, Null } internal class JsonReader { private static readonly IDictionary> parse_table; private Stack automaton_stack; private int current_input; private int current_symbol; private bool end_of_json; private bool end_of_input; private Lexer lexer; private bool parser_in_string; private bool parser_return; private bool read_started; private TextReader reader; private bool reader_is_owned; private bool skip_non_members; private object token_value; private JsonToken token; public bool AllowComments { get { return lexer.AllowComments; } set { lexer.AllowComments = value; } } public bool AllowSingleQuotedStrings { get { return lexer.AllowSingleQuotedStrings; } set { lexer.AllowSingleQuotedStrings = value; } } public bool SkipNonMembers { get { return skip_non_members; } set { skip_non_members = value; } } public bool EndOfInput => end_of_input; public bool EndOfJson => end_of_json; public JsonToken Token => token; public object Value => token_value; static JsonReader() { parse_table = PopulateParseTable(); } public JsonReader(string json_text) : this(new StringReader(json_text), owned: true) { } public JsonReader(TextReader reader) : this(reader, owned: false) { } private JsonReader(TextReader reader, bool owned) { if (reader == null) { throw new ArgumentNullException("reader"); } parser_in_string = false; parser_return = false; read_started = false; automaton_stack = new Stack(); automaton_stack.Push(65553); automaton_stack.Push(65543); lexer = new Lexer(reader); end_of_input = false; end_of_json = false; skip_non_members = true; this.reader = reader; reader_is_owned = owned; } private static IDictionary> PopulateParseTable() { IDictionary> result = new Dictionary>(); TableAddRow(result, ParserToken.Array); TableAddCol(result, ParserToken.Array, 91, 91, 65549); TableAddRow(result, ParserToken.ArrayPrime); TableAddCol(result, ParserToken.ArrayPrime, 34, 65550, 65551, 93); TableAddCol(result, ParserToken.ArrayPrime, 91, 65550, 65551, 93); TableAddCol(result, ParserToken.ArrayPrime, 93, 93); TableAddCol(result, ParserToken.ArrayPrime, 123, 65550, 65551, 93); TableAddCol(result, ParserToken.ArrayPrime, 65537, 65550, 65551, 93); TableAddCol(result, ParserToken.ArrayPrime, 65538, 65550, 65551, 93); TableAddCol(result, ParserToken.ArrayPrime, 65539, 65550, 65551, 93); TableAddCol(result, ParserToken.ArrayPrime, 65540, 65550, 65551, 93); TableAddRow(result, ParserToken.Object); TableAddCol(result, ParserToken.Object, 123, 123, 65545); TableAddRow(result, ParserToken.ObjectPrime); TableAddCol(result, ParserToken.ObjectPrime, 34, 65546, 65547, 125); TableAddCol(result, ParserToken.ObjectPrime, 125, 125); TableAddRow(result, ParserToken.Pair); TableAddCol(result, ParserToken.Pair, 34, 65552, 58, 65550); TableAddRow(result, ParserToken.PairRest); TableAddCol(result, ParserToken.PairRest, 44, 44, 65546, 65547); TableAddCol(result, ParserToken.PairRest, 125, 65554); TableAddRow(result, ParserToken.String); TableAddCol(result, ParserToken.String, 34, 34, 65541, 34); TableAddRow(result, ParserToken.Text); TableAddCol(result, ParserToken.Text, 91, 65548); TableAddCol(result, ParserToken.Text, 123, 65544); TableAddRow(result, ParserToken.Value); TableAddCol(result, ParserToken.Value, 34, 65552); TableAddCol(result, ParserToken.Value, 91, 65548); TableAddCol(result, ParserToken.Value, 123, 65544); TableAddCol(result, ParserToken.Value, 65537, 65537); TableAddCol(result, ParserToken.Value, 65538, 65538); TableAddCol(result, ParserToken.Value, 65539, 65539); TableAddCol(result, ParserToken.Value, 65540, 65540); TableAddRow(result, ParserToken.ValueRest); TableAddCol(result, ParserToken.ValueRest, 44, 44, 65550, 65551); TableAddCol(result, ParserToken.ValueRest, 93, 65554); return result; } private static void TableAddCol(IDictionary> parse_table, ParserToken row, int col, params int[] symbols) { parse_table[(int)row].Add(col, symbols); } private static void TableAddRow(IDictionary> parse_table, ParserToken rule) { parse_table.Add((int)rule, new Dictionary()); } private void ProcessNumber(string number) { int result2; long result3; ulong result4; if ((number.IndexOf('.') != -1 || number.IndexOf('e') != -1 || number.IndexOf('E') != -1) && double.TryParse(number, NumberStyles.Any, CultureInfo.InvariantCulture, out var result)) { token = JsonToken.Double; token_value = result; } else if (int.TryParse(number, NumberStyles.Integer, CultureInfo.InvariantCulture, out result2)) { token = JsonToken.Int; token_value = result2; } else if (long.TryParse(number, NumberStyles.Integer, CultureInfo.InvariantCulture, out result3)) { token = JsonToken.Long; token_value = result3; } else if (ulong.TryParse(number, NumberStyles.Integer, CultureInfo.InvariantCulture, out result4)) { token = JsonToken.Long; token_value = result4; } else { token = JsonToken.Int; token_value = 0; } } private void ProcessSymbol() { if (current_symbol == 91) { token = JsonToken.ArrayStart; parser_return = true; } else if (current_symbol == 93) { token = JsonToken.ArrayEnd; parser_return = true; } else if (current_symbol == 123) { token = JsonToken.ObjectStart; parser_return = true; } else if (current_symbol == 125) { token = JsonToken.ObjectEnd; parser_return = true; } else if (current_symbol == 34) { if (parser_in_string) { parser_in_string = false; parser_return = true; return; } if (token == JsonToken.None) { token = JsonToken.String; } parser_in_string = true; } else if (current_symbol == 65541) { token_value = lexer.StringValue; } else if (current_symbol == 65539) { token = JsonToken.Boolean; token_value = false; parser_return = true; } else if (current_symbol == 65540) { token = JsonToken.Null; parser_return = true; } else if (current_symbol == 65537) { ProcessNumber(lexer.StringValue); parser_return = true; } else if (current_symbol == 65546) { token = JsonToken.PropertyName; } else if (current_symbol == 65538) { token = JsonToken.Boolean; token_value = true; parser_return = true; } } private bool ReadToken() { if (end_of_input) { return false; } lexer.NextToken(); if (lexer.EndOfInput) { Close(); return false; } current_input = lexer.Token; return true; } public void Close() { if (end_of_input) { return; } end_of_input = true; end_of_json = true; if (reader_is_owned) { using (reader) { } } reader = null; } public bool Read() { if (end_of_input) { return false; } if (end_of_json) { end_of_json = false; automaton_stack.Clear(); automaton_stack.Push(65553); automaton_stack.Push(65543); } parser_in_string = false; parser_return = false; token = JsonToken.None; token_value = null; if (!read_started) { read_started = true; if (!ReadToken()) { return false; } } while (true) { if (parser_return) { if (automaton_stack.Peek() == 65553) { end_of_json = true; } return true; } current_symbol = automaton_stack.Pop(); ProcessSymbol(); if (current_symbol == current_input) { if (!ReadToken()) { break; } continue; } int[] array; try { array = parse_table[current_symbol][current_input]; } catch (KeyNotFoundException inner_exception) { throw new JsonException((ParserToken)current_input, inner_exception); } if (array[0] != 65554) { for (int num = array.Length - 1; num >= 0; num--) { automaton_stack.Push(array[num]); } } } if (automaton_stack.Peek() != 65553) { throw new JsonException("Input doesn't evaluate to proper JSON text"); } if (parser_return) { return true; } return false; } } internal enum Condition { InArray, InObject, NotAProperty, Property, Value } internal class WriterContext { public int Count; public bool InArray; public bool InObject; public bool ExpectingValue; public int Padding; } public class JsonWriter { private static readonly NumberFormatInfo number_format; private WriterContext context; private Stack ctx_stack; private bool has_reached_end; private char[] hex_seq; private int indentation; private int indent_value; private StringBuilder inst_string_builder; private bool pretty_print; private bool validate; private bool lower_case_properties; private TextWriter writer; public int IndentValue { get { return indent_value; } set { indentation = indentation / indent_value * value; indent_value = value; } } public bool PrettyPrint { get { return pretty_print; } set { pretty_print = value; } } public TextWriter TextWriter => writer; public bool Validate { get { return validate; } set { validate = value; } } public bool LowerCaseProperties { get { return lower_case_properties; } set { lower_case_properties = value; } } static JsonWriter() { number_format = NumberFormatInfo.InvariantInfo; } public JsonWriter() { inst_string_builder = new StringBuilder(); writer = new StringWriter(inst_string_builder); Init(); } public JsonWriter(StringBuilder sb) : this(new StringWriter(sb)) { } public JsonWriter(TextWriter writer) { if (writer == null) { throw new ArgumentNullException("writer"); } this.writer = writer; Init(); } private void DoValidation(Condition cond) { if (!context.ExpectingValue) { context.Count++; } if (!validate) { return; } if (has_reached_end) { throw new JsonException("A complete JSON symbol has already been written"); } switch (cond) { case Condition.InArray: if (!context.InArray) { throw new JsonException("Can't close an array here"); } break; case Condition.InObject: if (!context.InObject || context.ExpectingValue) { throw new JsonException("Can't close an object here"); } break; case Condition.NotAProperty: if (context.InObject && !context.ExpectingValue) { throw new JsonException("Expected a property"); } break; case Condition.Property: if (!context.InObject || context.ExpectingValue) { throw new JsonException("Can't add a property here"); } break; case Condition.Value: if (!context.InArray && (!context.InObject || !context.ExpectingValue)) { throw new JsonException("Can't add a value here"); } break; } } private void Init() { has_reached_end = false; hex_seq = new char[4]; indentation = 0; indent_value = 4; pretty_print = false; validate = true; lower_case_properties = false; ctx_stack = new Stack(); context = new WriterContext(); ctx_stack.Push(context); } private static void IntToHex(int n, char[] hex) { for (int i = 0; i < 4; i++) { int num = n % 16; if (num < 10) { hex[3 - i] = (char)(48 + num); } else { hex[3 - i] = (char)(65 + (num - 10)); } n >>= 4; } } private void Indent() { if (pretty_print) { indentation += indent_value; } } private void Put(string str) { if (pretty_print && !context.ExpectingValue) { for (int i = 0; i < indentation; i++) { writer.Write(' '); } } writer.Write(str); } private void PutNewline() { PutNewline(add_comma: true); } private void PutNewline(bool add_comma) { if (add_comma && !context.ExpectingValue && context.Count > 1) { writer.Write(','); } if (pretty_print && !context.ExpectingValue) { writer.Write(Environment.NewLine); } } private void PutString(string str) { Put(string.Empty); writer.Write('"'); int length = str.Length; for (int i = 0; i < length; i++) { switch (str[i]) { case '\n': writer.Write("\\n"); continue; case '\r': writer.Write("\\r"); continue; case '\t': writer.Write("\\t"); continue; case '"': case '\\': writer.Write('\\'); writer.Write(str[i]); continue; case '\f': writer.Write("\\f"); continue; case '\b': writer.Write("\\b"); continue; } if (str[i] >= ' ' && str[i] <= '~') { writer.Write(str[i]); continue; } IntToHex(str[i], hex_seq); writer.Write("\\u"); writer.Write(hex_seq); } writer.Write('"'); } private void Unindent() { if (pretty_print) { indentation -= indent_value; } } public override string ToString() { if (inst_string_builder == null) { return string.Empty; } return inst_string_builder.ToString(); } public void Reset() { has_reached_end = false; ctx_stack.Clear(); context = new WriterContext(); ctx_stack.Push(context); if (inst_string_builder != null) { inst_string_builder.Remove(0, inst_string_builder.Length); } } public void Write(bool boolean) { DoValidation(Condition.Value); PutNewline(); Put(boolean ? "true" : "false"); context.ExpectingValue = false; } public void Write(decimal number) { DoValidation(Condition.Value); PutNewline(); Put(Convert.ToString(number, number_format)); context.ExpectingValue = false; } public void Write(double number) { DoValidation(Condition.Value); PutNewline(); string text = Convert.ToString(number, number_format); Put(text); if (text.IndexOf('.') == -1 && text.IndexOf('E') == -1) { writer.Write(".0"); } context.ExpectingValue = false; } public void Write(float number) { DoValidation(Condition.Value); PutNewline(); string str = Convert.ToString(number, number_format); Put(str); context.ExpectingValue = false; } public void Write(int number) { DoValidation(Condition.Value); PutNewline(); Put(Convert.ToString(number, number_format)); context.ExpectingValue = false; } public void Write(long number) { DoValidation(Condition.Value); PutNewline(); Put(Convert.ToString(number, number_format)); context.ExpectingValue = false; } public void Write(string str) { DoValidation(Condition.Value); PutNewline(); if (str == null) { Put("null"); } else { PutString(str); } context.ExpectingValue = false; } [CLSCompliant(false)] public void Write(ulong number) { DoValidation(Condition.Value); PutNewline(); Put(Convert.ToString(number, number_format)); context.ExpectingValue = false; } public void WriteArrayEnd() { DoValidation(Condition.InArray); PutNewline(add_comma: false); ctx_stack.Pop(); if (ctx_stack.Count == 1) { has_reached_end = true; } else { context = ctx_stack.Peek(); context.ExpectingValue = false; } Unindent(); Put("]"); } public void WriteArrayStart() { DoValidation(Condition.NotAProperty); PutNewline(); Put("["); context = new WriterContext(); context.InArray = true; ctx_stack.Push(context); Indent(); } public void WriteObjectEnd() { DoValidation(Condition.InObject); PutNewline(add_comma: false); ctx_stack.Pop(); if (ctx_stack.Count == 1) { has_reached_end = true; } else { context = ctx_stack.Peek(); context.ExpectingValue = false; } Unindent(); Put("}"); } public void WriteObjectStart() { DoValidation(Condition.NotAProperty); PutNewline(); Put("{"); context = new WriterContext(); context.InObject = true; ctx_stack.Push(context); Indent(); } public void WritePropertyName(string property_name) { DoValidation(Condition.Property); PutNewline(); string text = ((property_name == null || !lower_case_properties) ? property_name : property_name.ToLowerInvariant()); PutString(text); if (pretty_print) { if (text.Length > context.Padding) { context.Padding = text.Length; } for (int num = context.Padding - text.Length; num >= 0; num--) { writer.Write(' '); } writer.Write(": "); } else { writer.Write(':'); } context.ExpectingValue = true; } } internal class FsmContext { public bool Return; public int NextState; public Lexer L; public int StateStack; } internal class Lexer { private delegate bool StateHandler(FsmContext ctx); private static readonly int[] fsm_return_table; private static readonly StateHandler[] fsm_handler_table; private bool allow_comments; private bool allow_single_quoted_strings; private bool end_of_input; private FsmContext fsm_context; private int input_buffer; private int input_char; private TextReader reader; private int state; private StringBuilder string_buffer; private string string_value; private int token; private int unichar; public bool AllowComments { get { return allow_comments; } set { allow_comments = value; } } public bool AllowSingleQuotedStrings { get { return allow_single_quoted_strings; } set { allow_single_quoted_strings = value; } } public bool EndOfInput => end_of_input; public int Token => token; public string StringValue => string_value; static Lexer() { PopulateFsmTables(out fsm_handler_table, out fsm_return_table); } public Lexer(TextReader reader) { allow_comments = true; allow_single_quoted_strings = true; input_buffer = 0; string_buffer = new StringBuilder(128); state = 1; end_of_input = false; this.reader = reader; fsm_context = new FsmContext(); fsm_context.L = this; } private static int HexValue(int digit) { switch (digit) { case 65: case 97: return 10; case 66: case 98: return 11; case 67: case 99: return 12; case 68: case 100: return 13; case 69: case 101: return 14; case 70: case 102: return 15; default: return digit - 48; } } private static void PopulateFsmTables(out StateHandler[] fsm_handler_table, out int[] fsm_return_table) { fsm_handler_table = new StateHandler[28] { State1, State2, State3, State4, State5, State6, State7, State8, State9, State10, State11, State12, State13, State14, State15, State16, State17, State18, State19, State20, State21, State22, State23, State24, State25, State26, State27, State28 }; fsm_return_table = new int[28] { 65542, 0, 65537, 65537, 0, 65537, 0, 65537, 0, 0, 65538, 0, 0, 0, 65539, 0, 0, 65540, 65541, 65542, 0, 0, 65541, 65542, 0, 0, 0, 0 }; } private static char ProcessEscChar(int esc_char) { switch (esc_char) { case 34: case 39: case 47: case 92: return Convert.ToChar(esc_char); case 110: return '\n'; case 116: return '\t'; case 114: return '\r'; case 98: return '\b'; case 102: return '\f'; default: return '?'; } } private static bool State1(FsmContext ctx) { while (ctx.L.GetChar()) { if (ctx.L.input_char == 32 || (ctx.L.input_char >= 9 && ctx.L.input_char <= 13)) { continue; } if (ctx.L.input_char >= 49 && ctx.L.input_char <= 57) { ctx.L.string_buffer.Append((char)ctx.L.input_char); ctx.NextState = 3; return true; } switch (ctx.L.input_char) { case 34: ctx.NextState = 19; ctx.Return = true; return true; case 44: case 58: case 91: case 93: case 123: case 125: ctx.NextState = 1; ctx.Return = true; return true; case 45: ctx.L.string_buffer.Append((char)ctx.L.input_char); ctx.NextState = 2; return true; case 48: ctx.L.string_buffer.Append((char)ctx.L.input_char); ctx.NextState = 4; return true; case 102: ctx.NextState = 12; return true; case 110: ctx.NextState = 16; return true; case 116: ctx.NextState = 9; return true; case 39: if (!ctx.L.allow_single_quoted_strings) { return false; } ctx.L.input_char = 34; ctx.NextState = 23; ctx.Return = true; return true; case 47: if (!ctx.L.allow_comments) { return false; } ctx.NextState = 25; return true; default: return false; } } return true; } private static bool State2(FsmContext ctx) { ctx.L.GetChar(); if (ctx.L.input_char >= 49 && ctx.L.input_char <= 57) { ctx.L.string_buffer.Append((char)ctx.L.input_char); ctx.NextState = 3; return true; } if (ctx.L.input_char == 48) { ctx.L.string_buffer.Append((char)ctx.L.input_char); ctx.NextState = 4; return true; } return false; } private static bool State3(FsmContext ctx) { while (ctx.L.GetChar()) { if (ctx.L.input_char >= 48 && ctx.L.input_char <= 57) { ctx.L.string_buffer.Append((char)ctx.L.input_char); continue; } if (ctx.L.input_char == 32 || (ctx.L.input_char >= 9 && ctx.L.input_char <= 13)) { ctx.Return = true; ctx.NextState = 1; return true; } switch (ctx.L.input_char) { case 44: case 93: case 125: ctx.L.UngetChar(); ctx.Return = true; ctx.NextState = 1; return true; case 46: ctx.L.string_buffer.Append((char)ctx.L.input_char); ctx.NextState = 5; return true; case 69: case 101: ctx.L.string_buffer.Append((char)ctx.L.input_char); ctx.NextState = 7; return true; default: return false; } } return true; } private static bool State4(FsmContext ctx) { ctx.L.GetChar(); if (ctx.L.input_char == 32 || (ctx.L.input_char >= 9 && ctx.L.input_char <= 13)) { ctx.Return = true; ctx.NextState = 1; return true; } switch (ctx.L.input_char) { case 44: case 93: case 125: ctx.L.UngetChar(); ctx.Return = true; ctx.NextState = 1; return true; case 46: ctx.L.string_buffer.Append((char)ctx.L.input_char); ctx.NextState = 5; return true; case 69: case 101: ctx.L.string_buffer.Append((char)ctx.L.input_char); ctx.NextState = 7; return true; default: return false; } } private static bool State5(FsmContext ctx) { ctx.L.GetChar(); if (ctx.L.input_char >= 48 && ctx.L.input_char <= 57) { ctx.L.string_buffer.Append((char)ctx.L.input_char); ctx.NextState = 6; return true; } return false; } private static bool State6(FsmContext ctx) { while (ctx.L.GetChar()) { if (ctx.L.input_char >= 48 && ctx.L.input_char <= 57) { ctx.L.string_buffer.Append((char)ctx.L.input_char); continue; } if (ctx.L.input_char == 32 || (ctx.L.input_char >= 9 && ctx.L.input_char <= 13)) { ctx.Return = true; ctx.NextState = 1; return true; } switch (ctx.L.input_char) { case 44: case 93: case 125: ctx.L.UngetChar(); ctx.Return = true; ctx.NextState = 1; return true; case 69: case 101: ctx.L.string_buffer.Append((char)ctx.L.input_char); ctx.NextState = 7; return true; default: return false; } } return true; } private static bool State7(FsmContext ctx) { ctx.L.GetChar(); if (ctx.L.input_char >= 48 && ctx.L.input_char <= 57) { ctx.L.string_buffer.Append((char)ctx.L.input_char); ctx.NextState = 8; return true; } int num = ctx.L.input_char; if (num == 43 || num == 45) { ctx.L.string_buffer.Append((char)ctx.L.input_char); ctx.NextState = 8; return true; } return false; } private static bool State8(FsmContext ctx) { while (ctx.L.GetChar()) { if (ctx.L.input_char >= 48 && ctx.L.input_char <= 57) { ctx.L.string_buffer.Append((char)ctx.L.input_char); continue; } if (ctx.L.input_char == 32 || (ctx.L.input_char >= 9 && ctx.L.input_char <= 13)) { ctx.Return = true; ctx.NextState = 1; return true; } int num = ctx.L.input_char; if (num == 44 || num == 93 || num == 125) { ctx.L.UngetChar(); ctx.Return = true; ctx.NextState = 1; return true; } return false; } return true; } private static bool State9(FsmContext ctx) { ctx.L.GetChar(); if (ctx.L.input_char == 114) { ctx.NextState = 10; return true; } return false; } private static bool State10(FsmContext ctx) { ctx.L.GetChar(); if (ctx.L.input_char == 117) { ctx.NextState = 11; return true; } return false; } private static bool State11(FsmContext ctx) { ctx.L.GetChar(); if (ctx.L.input_char == 101) { ctx.Return = true; ctx.NextState = 1; return true; } return false; } private static bool State12(FsmContext ctx) { ctx.L.GetChar(); if (ctx.L.input_char == 97) { ctx.NextState = 13; return true; } return false; } private static bool State13(FsmContext ctx) { ctx.L.GetChar(); if (ctx.L.input_char == 108) { ctx.NextState = 14; return true; } return false; } private static bool State14(FsmContext ctx) { ctx.L.GetChar(); if (ctx.L.input_char == 115) { ctx.NextState = 15; return true; } return false; } private static bool State15(FsmContext ctx) { ctx.L.GetChar(); if (ctx.L.input_char == 101) { ctx.Return = true; ctx.NextState = 1; return true; } return false; } private static bool State16(FsmContext ctx) { ctx.L.GetChar(); if (ctx.L.input_char == 117) { ctx.NextState = 17; return true; } return false; } private static bool State17(FsmContext ctx) { ctx.L.GetChar(); if (ctx.L.input_char == 108) { ctx.NextState = 18; return true; } return false; } private static bool State18(FsmContext ctx) { ctx.L.GetChar(); if (ctx.L.input_char == 108) { ctx.Return = true; ctx.NextState = 1; return true; } return false; } private static bool State19(FsmContext ctx) { while (ctx.L.GetChar()) { switch (ctx.L.input_char) { case 34: ctx.L.UngetChar(); ctx.Return = true; ctx.NextState = 20; return true; case 92: ctx.StateStack = 19; ctx.NextState = 21; return true; } ctx.L.string_buffer.Append((char)ctx.L.input_char); } return true; } private static bool State20(FsmContext ctx) { ctx.L.GetChar(); if (ctx.L.input_char == 34) { ctx.Return = true; ctx.NextState = 1; return true; } return false; } private static bool State21(FsmContext ctx) { ctx.L.GetChar(); switch (ctx.L.input_char) { case 117: ctx.NextState = 22; return true; case 34: case 39: case 47: case 92: case 98: case 102: case 110: case 114: case 116: ctx.L.string_buffer.Append(ProcessEscChar(ctx.L.input_char)); ctx.NextState = ctx.StateStack; return true; default: return false; } } private static bool State22(FsmContext ctx) { int num = 0; int num2 = 4096; ctx.L.unichar = 0; while (ctx.L.GetChar()) { if ((ctx.L.input_char >= 48 && ctx.L.input_char <= 57) || (ctx.L.input_char >= 65 && ctx.L.input_char <= 70) || (ctx.L.input_char >= 97 && ctx.L.input_char <= 102)) { ctx.L.unichar += HexValue(ctx.L.input_char) * num2; num++; num2 /= 16; if (num == 4) { ctx.L.string_buffer.Append(Convert.ToChar(ctx.L.unichar)); ctx.NextState = ctx.StateStack; return true; } continue; } return false; } return true; } private static bool State23(FsmContext ctx) { while (ctx.L.GetChar()) { switch (ctx.L.input_char) { case 39: ctx.L.UngetChar(); ctx.Return = true; ctx.NextState = 24; return true; case 92: ctx.StateStack = 23; ctx.NextState = 21; return true; } ctx.L.string_buffer.Append((char)ctx.L.input_char); } return true; } private static bool State24(FsmContext ctx) { ctx.L.GetChar(); if (ctx.L.input_char == 39) { ctx.L.input_char = 34; ctx.Return = true; ctx.NextState = 1; return true; } return false; } private static bool State25(FsmContext ctx) { ctx.L.GetChar(); switch (ctx.L.input_char) { case 42: ctx.NextState = 27; return true; case 47: ctx.NextState = 26; return true; default: return false; } } private static bool State26(FsmContext ctx) { while (ctx.L.GetChar()) { if (ctx.L.input_char == 10) { ctx.NextState = 1; return true; } } return true; } private static bool State27(FsmContext ctx) { while (ctx.L.GetChar()) { if (ctx.L.input_char == 42) { ctx.NextState = 28; return true; } } return true; } private static bool State28(FsmContext ctx) { while (ctx.L.GetChar()) { if (ctx.L.input_char != 42) { if (ctx.L.input_char == 47) { ctx.NextState = 1; return true; } ctx.NextState = 27; return true; } } return true; } private bool GetChar() { if ((input_char = NextChar()) != -1) { return true; } end_of_input = true; return false; } private int NextChar() { if (input_buffer != 0) { int result = input_buffer; input_buffer = 0; return result; } return reader.Read(); } public bool NextToken() { fsm_context.Return = false; while (true) { if (!fsm_handler_table[state - 1](fsm_context)) { throw new JsonException(input_char); } if (end_of_input) { return false; } if (fsm_context.Return) { break; } state = fsm_context.NextState; } string_value = string_buffer.ToString(); string_buffer.Remove(0, string_buffer.Length); token = fsm_return_table[state - 1]; if (token == 65542) { token = input_char; } state = fsm_context.NextState; return true; } private void UngetChar() { input_buffer = input_char; } } internal enum ParserToken { None = 65536, Number, True, False, Null, CharSeq, Char, Text, Object, ObjectPrime, Pair, PairRest, Array, ArrayPrime, Value, ValueRest, String, End, Epsilon } }