src/Bannerlord.LauncherEx/ResourceManagers/WidgetFactoryManager.cs
using Bannerlord.LauncherEx.Patches;
using HarmonyLib;
using HarmonyLib.BUTR.Extensions;
using System;
using System.Collections;
using System.Collections.Concurrent;
using System.Collections.Generic;
using System.Linq;
using System.Xml;
using TaleWorlds.GauntletUI;
using TaleWorlds.GauntletUI.BaseTypes;
using TaleWorlds.GauntletUI.PrefabSystem;
namespace Bannerlord.LauncherEx.ResourceManagers;
/// <summary>
/// https://github.com/Aragas/Bannerlord.MBOptionScreen/blob/dev/src/MCM.UI/Patches/WidgetFactoryManager.cs
/// </summary>
internal static class WidgetFactoryManager
{
private delegate void ReloadDelegate();
private static readonly ReloadDelegate? Reload =
AccessTools2.GetDeclaredDelegate<ReloadDelegate>(typeof(WidgetInfo), "Reload");
private static readonly AccessTools.FieldRef<WidgetFactory, IDictionary>? _liveCustomTypes =
AccessTools2.FieldRefAccess<WidgetFactory, IDictionary>("_liveCustomTypes");
private delegate Widget WidgetConstructor(UIContext uiContext);
private static readonly ConcurrentDictionary<Type, WidgetConstructor?> WidgetConstructors = new();
private static readonly Dictionary<string, Func<WidgetPrefab?>> CustomTypes = new();
private static readonly Dictionary<string, Type> BuiltinTypes = new();
private static readonly Dictionary<string, WidgetPrefab> LiveCustomTypes = new();
private static readonly Dictionary<string, int> LiveInstanceTracker = new();
private static Harmony? _harmony;
private static WeakReference<WidgetFactory?> WidgetFactoryReference { get; } = new(null);
public static void SetWidgetFactory(WidgetFactory widgetFactory)
{
WidgetFactoryReference.SetTarget(widgetFactory);
}
public static WidgetPrefab? Create(string name, XmlDocument doc)
{
if (!WidgetFactoryReference.TryGetTarget(out var widgetFactory) || widgetFactory is null)
return null;
return WidgetPrefabPatch.LoadFromDocument(
widgetFactory.PrefabExtensionContext,
widgetFactory.WidgetAttributeContext,
name,
doc);
}
public static void Register(Type widgetType)
{
if (Reload is null) return;
BuiltinTypes[widgetType.Name] = widgetType;
Reload();
}
public static void Register(string name, Func<WidgetPrefab?> create) => CustomTypes.Add(name, create);
public static void CreateAndRegister(string name, XmlDocument xmlDocument) => Register(name, () => Create($"{name}.xml", xmlDocument));
public static void Clear()
{
WidgetConstructors.Clear();
CustomTypes.Clear();
BuiltinTypes.Clear();
LiveCustomTypes.Clear();
LiveInstanceTracker.Clear();
WidgetFactoryReference.SetTarget(null);
}
public static bool Enable(Harmony harmony)
{
_harmony = harmony;
var res1 = harmony.TryPatch(
AccessTools2.Method(typeof(WidgetFactory), "GetCustomType"),
prefix: AccessTools2.DeclaredMethod(typeof(WidgetFactoryManager), nameof(GetCustomTypePrefix)));
if (!res1) return false;
var res2 = harmony.TryPatch(
AccessTools2.Method(typeof(WidgetFactory), "GetWidgetTypes"),
prefix: AccessTools2.DeclaredMethod(typeof(WidgetFactoryManager), nameof(GetWidgetTypesPostfix)));
if (!res2) return false;
var res3 = harmony.TryPatch(
AccessTools2.Method(typeof(WidgetFactory), "IsCustomType"),
prefix: AccessTools2.DeclaredMethod(typeof(WidgetFactoryManager), nameof(IsCustomTypePrefix)));
if (!res3) return false;
var res4 = harmony.TryPatch(
AccessTools2.Method(typeof(WidgetFactory), "OnUnload"),
prefix: AccessTools2.DeclaredMethod(typeof(WidgetFactoryManager), nameof(OnUnloadPrefix)));
if (!res4) return false;
var res5 = harmony.TryPatch(
AccessTools2.Method(typeof(WidgetFactory), "Initialize"),
prefix: AccessTools2.DeclaredMethod(typeof(WidgetFactoryManager), nameof(InitializePostfix)));
if (!res5) return false;
var res6 = harmony.TryPatch(
AccessTools2.Method(typeof(WidgetFactory), "CreateBuiltinWidget"),
prefix: AccessTools2.DeclaredMethod(typeof(WidgetFactoryManager), nameof(CreateBuiltinWidgetPrefix)));
if (!res6) return false;
return true;
}
private static void GetWidgetTypesPostfix(ref IEnumerable<string> __result)
{
__result = __result.Concat(BuiltinTypes.Keys).Concat(CustomTypes.Keys);
}
private static bool CreateBuiltinWidgetPrefix(UIContext context, string typeName, ref object? __result)
{
if (!BuiltinTypes.TryGetValue(typeName, out var type))
return true;
var ctor = WidgetConstructors.GetOrAdd(type, static x => AccessTools2.GetDeclaredConstructorDelegate<WidgetConstructor>(x, new[] { typeof(UIContext) }));
if (ctor is null)
return true;
__result = ctor(context);
return false;
}
private static bool IsCustomTypePrefix(string typeName, ref bool __result)
{
if (!CustomTypes.ContainsKey(typeName))
return true;
__result = true;
return false;
}
private static bool GetCustomTypePrefix(WidgetFactory __instance, string typeName, ref WidgetPrefab __result)
{
if (_liveCustomTypes?.Invoke(__instance) is { } ____liveCustomTypes && ____liveCustomTypes.Contains(typeName) || !CustomTypes.ContainsKey(typeName))
return true;
if (LiveCustomTypes.TryGetValue(typeName, out var liveWidgetPrefab))
{
LiveInstanceTracker[typeName]++;
__result = liveWidgetPrefab;
return false;
}
if (CustomTypes[typeName]?.Invoke() is { } widgetPrefab)
{
LiveCustomTypes.Add(typeName, widgetPrefab);
LiveInstanceTracker[typeName] = 1;
__result = widgetPrefab;
return false;
}
return true;
}
private static bool OnUnloadPrefix(string typeName)
{
if (LiveCustomTypes.ContainsKey(typeName))
{
LiveInstanceTracker[typeName]--;
if (LiveInstanceTracker[typeName] == 0)
{
LiveCustomTypes.Remove(typeName);
LiveInstanceTracker.Remove(typeName);
}
return false;
}
return true;
}
private static void InitializePostfix(ref WidgetFactory __instance)
{
SetWidgetFactory(__instance);
_harmony?.Unpatch(
AccessTools2.Method(typeof(WidgetFactory), "Initialize"),
AccessTools2.DeclaredMethod(typeof(WidgetFactoryManager), nameof(InitializePostfix)));
}
private static IEnumerable<CodeInstruction> BlankTranspiler(IEnumerable<CodeInstruction> instructions) => instructions;
}