ClientMods/CustomInteractions/CustomInteractions.cs

174 lines
7.8 KiB
C#
Raw Normal View History

using Comfort.Common;
using EFT.InventoryLogic;
using EFT.UI;
using JetBrains.Annotations;
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Linq;
using System.Reflection;
using UnityEngine;
using EmptyInteractionsAbstractClass = GClass3061;
2024-03-05 22:48:21 +02:00
namespace IcyClawz.CustomInteractions;
[EditorBrowsable(EditorBrowsableState.Never)]
public interface ICustomInteractionsProvider { } // Do not implement this directly
public interface IItemCustomInteractionsProvider : ICustomInteractionsProvider
{
IEnumerable<CustomInteraction> GetCustomInteractions(ItemUiContext uiContext, EItemViewType viewType, Item item);
}
public static class CustomInteractionsManager
{
2024-03-05 22:48:21 +02:00
internal static readonly LinkedList<ICustomInteractionsProvider> Providers = [];
2024-03-05 22:48:21 +02:00
public static void Register(ICustomInteractionsProvider provider)
{
2024-03-05 22:48:21 +02:00
if (!Providers.Contains(provider))
Providers.AddLast(provider);
}
2024-03-05 22:48:21 +02:00
}
2024-03-05 22:48:21 +02:00
public class CustomInteraction()
{
internal readonly CustomInteractionImpl Impl = new(UnityEngine.Random.Range(0, int.MaxValue).ToString("x4"));
2024-03-05 22:48:21 +02:00
public Func<string> Caption { get => Impl.Caption; set => Impl.Caption = value; }
public Func<Sprite> Icon { get => Impl.Icon; set => Impl.Icon = value; }
public Action Action { get => Impl.Action; set => Impl.Action = value; }
public Func<CustomSubInteractions> SubMenu { get => Impl.SubMenu; set => Impl.SubMenu = value; }
public Func<bool> Enabled { get => Impl.Enabled; set => Impl.Enabled = value; }
public Func<string> Error { get => Impl.Error; set => Impl.Error = value; }
}
internal sealed class CustomInteractionImpl(string id) : DynamicInteractionClass(id, id)
2024-03-05 22:48:21 +02:00
{
public Func<string> Caption { get; set; }
public new Func<Sprite> Icon { get; set; }
public Action Action { get => action_0; set => action_0 = value; }
public Func<CustomSubInteractions> SubMenu { get; set; }
public Func<bool> Enabled { get; set; }
public Func<string> Error { get; set; }
2024-03-05 22:48:21 +02:00
public bool IsInteractive() => Enabled?.Invoke() ?? true;
}
2024-03-05 22:48:21 +02:00
public abstract class CustomSubInteractions(ItemUiContext uiContext)
{
internal readonly CustomSubInteractionsImpl Impl = new(uiContext);
2024-03-05 22:48:21 +02:00
public bool ExaminationRequired
{
2024-03-05 22:48:21 +02:00
get => Impl.ExaminationRequiredInternal;
set => Impl.ExaminationRequiredInternal = value;
}
2024-03-05 22:48:21 +02:00
public void Add(CustomInteraction interaction) => Impl.AddCustomInteraction(interaction);
public void AddRange(IEnumerable<CustomInteraction> interactions) => interactions.ExecuteForEach(Impl.AddCustomInteraction);
public void Remove(CustomInteraction interaction) => Impl.RemoveCustomInteraction(interaction);
public void RemoveRange(IEnumerable<CustomInteraction> interactions) => interactions.ExecuteForEach(Impl.RemoveCustomInteraction);
}
internal sealed class CustomSubInteractionsImpl(ItemUiContext uiContext) : EmptyInteractionsAbstractClass(uiContext)
2024-03-05 22:48:21 +02:00
{
public IEnumerable<CustomInteractionImpl> CustomInteractions => DynamicInteractions.OfType<CustomInteractionImpl>();
2024-03-05 22:48:21 +02:00
public bool ExaminationRequiredInternal { get; set; } = true;
public override bool ExaminationRequired => ExaminationRequiredInternal;
public override bool HasIcons => CustomInteractions.Any(interaction => interaction.Icon is not null);
}
2024-03-05 22:48:21 +02:00
internal static class AbstractInteractionsExtensions
{
private static Dictionary<string, DynamicInteractionClass> GetDynamicInteractions<T>(this ItemInfoInteractionsAbstractClass<T> instance) where T : struct, Enum =>
typeof(ItemInfoInteractionsAbstractClass<T>).GetField("dictionary_0", BindingFlags.NonPublic | BindingFlags.Instance)
.GetValue(instance) as Dictionary<string, DynamicInteractionClass>;
public static void AddCustomInteraction<T>(this ItemInfoInteractionsAbstractClass<T> instance, CustomInteraction interaction) where T : struct, Enum =>
2024-03-05 22:48:21 +02:00
instance.GetDynamicInteractions()[interaction.Impl.Key] = interaction.Impl;
public static void RemoveCustomInteraction<T>(this ItemInfoInteractionsAbstractClass<T> instance, CustomInteraction interaction) where T : struct, Enum =>
2024-03-05 22:48:21 +02:00
instance.GetDynamicInteractions().Remove(interaction.Impl.Key);
}
2024-03-05 22:48:21 +02:00
internal static class InteractionButtonsContainerExtensions
{
private static readonly FieldInfo ButtonsContainerField =
typeof(InteractionButtonsContainer).GetField("_buttonsContainer", BindingFlags.NonPublic | BindingFlags.Instance);
2024-03-05 22:48:21 +02:00
private static RectTransform GetButtonsContainer(this InteractionButtonsContainer instance) =>
ButtonsContainerField.GetValue(instance) as RectTransform;
2024-03-05 22:48:21 +02:00
private static readonly FieldInfo ButtonTemplateField =
typeof(InteractionButtonsContainer).GetField("_buttonTemplate", BindingFlags.NonPublic | BindingFlags.Instance);
2024-03-05 22:48:21 +02:00
private static SimpleContextMenuButton GetButtonTemplate(this InteractionButtonsContainer instance) =>
ButtonTemplateField.GetValue(instance) as SimpleContextMenuButton;
2024-03-05 22:48:21 +02:00
private static readonly FieldInfo CurrentButtonField =
typeof(InteractionButtonsContainer).GetField("simpleContextMenuButton_0", BindingFlags.NonPublic | BindingFlags.Instance);
2024-03-05 22:48:21 +02:00
private static void SetCurrentButton(this InteractionButtonsContainer instance, SimpleContextMenuButton button) =>
CurrentButtonField.SetValue(instance, button);
2024-03-05 22:48:21 +02:00
private static readonly MethodInfo CreateButtonMethod =
typeof(InteractionButtonsContainer).GetMethod("method_1", BindingFlags.Public | BindingFlags.Instance);
2024-03-05 22:48:21 +02:00
private static SimpleContextMenuButton CreateButton(this InteractionButtonsContainer instance,
string key, string caption, SimpleContextMenuButton template, RectTransform container,
[CanBeNull] Sprite sprite, [CanBeNull] Action onButtonClicked, [CanBeNull] Action onMouseHover,
bool subMenu = false, bool autoClose = true) =>
(SimpleContextMenuButton)CreateButtonMethod.Invoke(instance, [
key, caption, template, container, sprite, onButtonClicked, onMouseHover, subMenu, autoClose
]);
2024-03-05 22:48:21 +02:00
private static readonly MethodInfo CloseSubMenuMethod =
typeof(InteractionButtonsContainer).GetMethod("method_4", BindingFlags.Public | BindingFlags.Instance);
2024-03-05 22:48:21 +02:00
private static void CloseSubMenu(this InteractionButtonsContainer instance) =>
CloseSubMenuMethod.Invoke(instance, null);
private static readonly MethodInfo AddButtonMethod =
typeof(InteractionButtonsContainer).GetMethod("method_5", BindingFlags.Public | BindingFlags.Instance);
2024-03-05 22:48:21 +02:00
private static void AddButton(this InteractionButtonsContainer instance, SimpleContextMenuButton button) =>
AddButtonMethod.Invoke(instance, [button]);
public static void AddCustomButton(this InteractionButtonsContainer instance, CustomInteractionImpl impl)
{
2024-03-05 22:48:21 +02:00
bool isInteractive = impl.IsInteractive();
SimpleContextMenuButton button = null;
button = instance.CreateButton(
impl.Key,
impl.Caption?.Invoke() ?? "",
instance.GetButtonTemplate(),
instance.GetButtonsContainer(),
impl.Icon?.Invoke(),
() =>
{
if (isInteractive)
impl.Execute();
},
() =>
{
instance.SetCurrentButton(button);
instance.CloseSubMenu();
if (isInteractive)
{
2024-03-05 22:48:21 +02:00
CustomSubInteractions subMenu = impl.SubMenu?.Invoke();
if (subMenu is not null)
instance.SetSubInteractions(subMenu.Impl);
}
},
impl.SubMenu is not null,
false
);
button.SetButtonInteraction(
isInteractive ? SuccessfulResult.New : new FailedResult(impl.Error?.Invoke() ?? "", 0)
2024-03-05 22:48:21 +02:00
);
instance.AddButton(button);
}
}