AxibugEmuOnline/AxibugEmuOnline.Client/Assets/Script/UI/OptionUI/OptionUI.cs

432 lines
13 KiB
C#
Raw Normal View History

2024-09-12 15:08:48 +08:00
using DG.Tweening;
using System;
using System.Collections.Generic;
2024-11-13 14:03:11 +08:00
using System.Linq;
2024-09-12 15:08:48 +08:00
using UnityEngine;
namespace AxibugEmuOnline.Client
{
public class OptionUI : CommandExecuter
{
[SerializeField]
RectTransform MenuRoot;
2024-09-12 17:47:05 +08:00
[SerializeField]
Selector SelectBorder;
2024-09-12 15:08:48 +08:00
[Space]
[Header("ģ<><C4A3>")]
[SerializeField] OptionUI_ExecuteItem TEMPLATE_EXECUTEITEM;
private OptionUI m_child;
private OptionUI m_parent;
public override bool AloneMode => true;
public override bool Enable => m_bPoped && (m_child == null || !m_child.m_bPoped);
2024-09-12 15:08:48 +08:00
private bool m_bPoped = false;
2024-09-13 10:28:02 +08:00
private List<OptionUI_MenuItem> m_runtimeMenuItems = new List<OptionUI_MenuItem>();
2024-09-12 15:08:48 +08:00
2024-09-12 17:47:05 +08:00
private int m_selectIndex = -1;
public int SelectIndex
{
get { return m_selectIndex; }
set
{
value = Mathf.Clamp(value, 0, m_runtimeMenuItems.Count - 1);
2024-09-12 17:47:05 +08:00
if (m_selectIndex == value) return;
2024-11-13 14:03:11 +08:00
var gap = value - m_selectIndex;
while (!m_runtimeMenuItems[value].Visible)
{
var temp = value;
if (gap > 0)
{
temp++;
}
else
{
temp--;
}
if (temp >= 0 && temp < m_runtimeMenuItems.Count)
value = temp;
}
2024-09-12 17:47:05 +08:00
m_selectIndex = value;
OptionUI_MenuItem optionUI_MenuItem = m_runtimeMenuItems[m_selectIndex];
2024-09-13 10:28:02 +08:00
optionUI_MenuItem.OnFocus();
var itemUIRect = optionUI_MenuItem.transform as RectTransform;
SelectBorder.Target = itemUIRect;
2024-09-12 17:47:05 +08:00
}
}
2024-09-12 15:08:48 +08:00
protected override void Awake()
{
TEMPLATE_EXECUTEITEM.gameObject.SetActiveEx(false);
2024-09-12 17:47:05 +08:00
SelectBorder.gameObject.SetActiveEx(false);
2024-09-12 15:08:48 +08:00
base.Awake();
}
2024-09-14 15:32:29 +08:00
protected override void Update()
{
UpdateMenuState();
2024-11-27 13:30:56 +08:00
base.Update();
2024-09-14 15:32:29 +08:00
}
private void UpdateMenuState()
{
bool dirty = false;
dirty = checkDirty();
2024-09-14 15:32:29 +08:00
if (dirty)
{
RebuildSelectIndex();
}
}
2024-11-13 13:55:36 +08:00
private void RebuildSelectIndex()
{
Canvas.ForceUpdateCanvases();
SelectIndex = Mathf.Clamp(SelectIndex, 0, m_runtimeMenuItems.Count - 1);
var selectItem = m_runtimeMenuItems[SelectIndex];
if (selectItem.Visible == false)
{
bool find = false;
int currentSelect = SelectIndex;
while (currentSelect > 0)
2024-09-14 15:32:29 +08:00
{
currentSelect--;
if (m_runtimeMenuItems[currentSelect].Visible)
2024-09-14 15:32:29 +08:00
{
find = true;
2024-09-14 15:32:29 +08:00
}
}
if (!find)
{
currentSelect = SelectIndex;
while (currentSelect < m_runtimeMenuItems.Count)
2024-09-14 15:32:29 +08:00
{
if (m_runtimeMenuItems[currentSelect].Visible)
2024-09-14 15:32:29 +08:00
{
find = true;
2024-09-14 15:32:29 +08:00
}
currentSelect++;
2024-09-14 15:32:29 +08:00
}
}
if (find)
SelectIndex = currentSelect;
}
else
{
var itemUIRect = selectItem.transform as RectTransform;
SelectBorder.Target = itemUIRect;
}
}
private bool checkDirty()
{
bool dirty = false;
foreach (var menuItem in m_runtimeMenuItems)
{
if (menuItem.gameObject.activeSelf != menuItem.Visible)
2024-11-13 13:55:36 +08:00
{
dirty = true;
menuItem.gameObject.SetActive(menuItem.Visible);
2024-11-13 13:55:36 +08:00
}
2024-09-14 15:32:29 +08:00
}
return dirty;
2024-09-14 15:32:29 +08:00
}
2024-11-21 20:32:41 +08:00
IKeyMapperChanger m_lastCS;
2024-11-28 16:33:00 +08:00
private Action m_onClose;
/// <summary>
/// <20><><EFBFBD>˵<EFBFBD><CBB5><EFBFBD><EFBFBD><EFBFBD>ʱ,<2C><>̬<EFBFBD><CCAC><EFBFBD><EFBFBD>һ<EFBFBD><D2BB><EFBFBD>˵<EFBFBD>ѡ<EFBFBD><D1A1>
/// </summary>
/// <param name="menu"></param>
public void AddOptionMenuWhenPoping(OptionMenu menu)
2024-09-12 15:08:48 +08:00
{
2024-11-28 16:33:00 +08:00
if (!m_bPoped) return;
CreateRuntimeMenuItem(menu);
Canvas.ForceUpdateCanvases();
OptionUI_MenuItem optionUI_MenuItem = m_runtimeMenuItems[m_selectIndex];
SelectBorder.Target = optionUI_MenuItem.transform as RectTransform;
2024-11-28 16:33:00 +08:00
}
public void Pop<T>(List<T> menus, int defaultIndex = 0, Action onClose = null) where T : OptionMenu
{
m_onClose = onClose;
2024-09-12 15:08:48 +08:00
ReleaseRuntimeMenus();
foreach (var menu in menus) CreateRuntimeMenuItem(menu);
CommandDispatcher.Instance.RegistController(this);
2024-09-12 17:47:05 +08:00
SelectBorder.gameObject.SetActiveEx(true);
Canvas.ForceUpdateCanvases();
2024-11-14 19:25:44 +08:00
m_selectIndex = defaultIndex;
2024-09-13 10:28:02 +08:00
OptionUI_MenuItem optionUI_MenuItem = m_runtimeMenuItems[defaultIndex];
optionUI_MenuItem.OnFocus();
2024-11-28 16:33:00 +08:00
2024-09-13 10:28:02 +08:00
var itemUIRect = optionUI_MenuItem.transform as RectTransform;
SelectBorder.Target = itemUIRect;
2024-09-12 17:47:05 +08:00
2024-09-12 15:08:48 +08:00
if (!m_bPoped)
{
m_bPoped = true;
Vector2 start = new Vector2(0, MenuRoot.anchoredPosition.y);
Vector2 end = new Vector2(-MenuRoot.rect.width, MenuRoot.anchoredPosition.y);
2024-09-12 15:08:48 +08:00
DOTween.To(
() => start,
2024-09-12 15:08:48 +08:00
(value) =>
{
var moveDelta = value - start;
start = value;
var topParent = m_parent;
while (topParent != null && topParent.m_parent != null)
{
topParent = topParent.m_parent;
}
if (topParent != null)
{
topParent.MenuRoot.anchoredPosition += moveDelta;
}
else
{
MenuRoot.anchoredPosition += moveDelta;
}
2024-09-12 15:08:48 +08:00
},
end,
2024-09-12 15:08:48 +08:00
0.3f
).SetEase(Ease.OutCubic);
2024-09-14 17:22:01 +08:00
2024-11-21 20:32:41 +08:00
m_lastCS = CommandDispatcher.Instance.Current;
CommandDispatcher.Instance.Current = CommandDispatcher.Instance.Normal;
2024-09-12 15:08:48 +08:00
}
2024-09-23 18:31:05 +08:00
2024-09-12 15:08:48 +08:00
}
public void Hide()
{
if (m_bPoped)
{
2024-09-18 15:53:58 +08:00
ReleaseRuntimeMenus();
2024-09-14 17:22:01 +08:00
m_runtimeMenuItems.Clear();
2024-09-12 17:47:05 +08:00
SelectBorder.gameObject.SetActiveEx(false);
CommandDispatcher.Instance.UnRegistController(this);
2024-09-12 15:08:48 +08:00
Canvas.ForceUpdateCanvases();
Vector2 start = new Vector2(-MenuRoot.rect.width, MenuRoot.anchoredPosition.y);
Vector2 end = new Vector2(0, MenuRoot.anchoredPosition.y);
2024-09-12 15:08:48 +08:00
DOTween.To(
() => start,
2024-09-12 15:08:48 +08:00
(value) =>
{
var moveDelta = value - start;
start = value;
var topParent = m_parent;
while (topParent != null && topParent.m_parent != null)
{
topParent = topParent.m_parent;
}
if (topParent != null)
{
topParent.MenuRoot.anchoredPosition += moveDelta;
}
else
{
MenuRoot.anchoredPosition += moveDelta;
}
2024-09-12 15:08:48 +08:00
},
end,
2024-09-12 15:08:48 +08:00
0.3f
).SetEase(Ease.OutCubic);
2024-09-14 17:22:01 +08:00
m_bPoped = false;
2024-09-23 18:31:05 +08:00
2024-11-21 20:32:41 +08:00
CommandDispatcher.Instance.Current = m_lastCS;
2024-11-28 16:33:00 +08:00
m_onClose?.Invoke();
m_onClose = null;
2024-09-12 15:08:48 +08:00
}
}
private void CreateRuntimeMenuItem(OptionMenu menuData)
{
if (menuData is ExecuteMenu executeMenu)
{
var menuUI = GameObject.Instantiate(TEMPLATE_EXECUTEITEM.gameObject, TEMPLATE_EXECUTEITEM.transform.parent).GetComponent<OptionUI_ExecuteItem>();
menuUI.gameObject.SetActive(true);
menuUI.SetData(this, executeMenu);
2024-09-12 15:08:48 +08:00
m_runtimeMenuItems.Add(menuUI);
}
2024-11-13 13:55:36 +08:00
else
{
throw new NotImplementedException($"<22>ݲ<EFBFBD>֧<EFBFBD>ֵIJ˵<C4B2><CBB5><EFBFBD><EFBFBD><EFBFBD>{menuData.GetType().Name}");
}
2024-09-12 15:08:48 +08:00
}
private void ReleaseRuntimeMenus()
{
foreach (var item in m_runtimeMenuItems)
{
item.OnHide();
2024-09-12 15:08:48 +08:00
Destroy(item.gameObject);
}
m_runtimeMenuItems.Clear();
}
2024-09-12 17:47:05 +08:00
protected override void OnCmdSelectItemDown()
2024-09-12 15:08:48 +08:00
{
2024-09-12 17:47:05 +08:00
SelectIndex++;
}
2024-09-12 17:47:05 +08:00
protected override void OnCmdSelectItemUp()
{
SelectIndex--;
2024-09-12 15:08:48 +08:00
}
2024-09-13 10:28:02 +08:00
protected override void OnCmdBack()
{
Hide();
}
protected override void OnCmdSelectItemRight()
{
var executer = m_runtimeMenuItems[SelectIndex];
if (!executer.IsExpandMenu) return;
OnCmdEnter();
}
2024-09-13 10:28:02 +08:00
protected override bool OnCmdEnter()
{
2024-11-08 16:23:26 +08:00
var executer = m_runtimeMenuItems[SelectIndex];
2024-11-28 16:33:00 +08:00
bool cancelHide = false;
executer.OnExecute(this, ref cancelHide);
if (!cancelHide) Hide();
return false;
2024-09-13 10:28:02 +08:00
}
2024-09-12 15:08:48 +08:00
/// <summary>
/// չ<><D5B9><EFBFBD>¼<EFBFBD><C2BC>˵<EFBFBD>
/// </summary>
/// <typeparam name="T"></typeparam>
/// <param name="menus"></param>
/// <param name="defaultIndex"></param>
/// <param name="onClose"></param>
public void ExpandSubMenu<T>(List<T> menus, int defaultIndex = 0, Action onClose = null) where T : OptionMenu
{
if (m_child == null)
{
var sourcePrefab = Resources.Load<GameObject>("UIPrefabs/OptionUI");
m_child = Instantiate(sourcePrefab, transform).GetComponent<OptionUI>();
m_child.name = $"{name}_Sub";
m_child.m_parent = this;
}
Canvas.ForceUpdateCanvases();
m_child.Pop(menus, 0, onClose);
}
public void RemoveItem(OptionUI_MenuItem ui)
{
var index = m_runtimeMenuItems.IndexOf(ui);
if (index == -1) return;
2024-09-13 17:39:48 +08:00
m_runtimeMenuItems.Remove(ui);
ui.OnHide();
Destroy(ui.gameObject);
RebuildSelectIndex();
}
}
2024-09-12 15:08:48 +08:00
2024-11-28 16:33:00 +08:00
/// <summary>
/// <20><><EFBFBD><EFBFBD>ִ<EFBFBD><D6B4><EFBFBD><EFBFBD>Ϊ<EFBFBD>IJ˵<C4B2>
/// </summary>
2024-11-13 13:55:36 +08:00
public abstract class ExecuteMenu : OptionMenu
2024-09-12 15:08:48 +08:00
{
public ExecuteMenu(string name, Sprite icon = null) : base(name, icon) { }
2024-11-28 16:33:00 +08:00
public abstract void OnExcute(OptionUI optionUI, ref bool cancelHide);
2024-09-12 15:08:48 +08:00
}
/// <summary>
/// <20><><EFBFBD><EFBFBD>չ<EFBFBD><D5B9><EFBFBD><EFBFBD>Ϊ<EFBFBD>IJ˵<C4B2>
/// </summary>
public abstract class ExpandMenu : ExecuteMenu
{
protected ExpandMenu(string name, Sprite icon = null) : base(name, icon) { }
public sealed override void OnExcute(OptionUI optionUI, ref bool cancelHide)
{
cancelHide = true;
optionUI.ExpandSubMenu(GetOptionMenus());
}
protected abstract List<OptionMenu> GetOptionMenus();
}
2024-11-28 16:33:00 +08:00
/// <summary>
/// <20><><EFBFBD><EFBFBD>ֵ<EFBFBD><D6B5><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ʾ<EFBFBD>ͱ༭<CDB1>IJ˵<C4B2>
/// </summary>
/// <typeparam name="T"></typeparam>
2024-09-12 15:08:48 +08:00
public class ValueSetMenu<T> : ValueSetMenu
{
public sealed override Type ValueType => typeof(T);
public T Value { get; private set; }
public sealed override object ValueRaw => Value;
public sealed override void OnValueChanged(object newValue)
{
Value = (T)newValue;
}
protected ValueSetMenu(string name) : base(name) { }
}
2024-11-14 17:14:52 +08:00
2024-11-28 16:33:00 +08:00
/// <summary> <20><>Ҫֱ<D2AA>Ӽ̳<D3BC><CCB3><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> </summary>
public abstract class OptionMenu
{
public string Name { get; protected set; }
public Sprite Icon { get; protected set; }
public virtual bool Visible => true;
public virtual bool Enable => true;
public OptionMenu(string name, Sprite icon = null)
{
Name = name;
Icon = icon;
}
public virtual void OnFocus() { }
public virtual void OnShow(OptionUI_MenuItem ui) { }
public virtual void OnHide() { }
2024-11-28 16:33:00 +08:00
}
/// <summary> <20><>Ҫֱ<D2AA>Ӽ̳<D3BC><CCB3><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> </summary>
2024-11-14 17:14:52 +08:00
public abstract class ValueSetMenu : OptionMenu
{
public ValueSetMenu(string name) : base(name) { }
public abstract Type ValueType { get; }
public abstract object ValueRaw { get; }
public abstract void OnValueChanged(object newValue);
}
2024-09-12 15:08:48 +08:00
}