Files
BABA_YAGA/Assets/Scripts/UI/MainMenuController.cs

193 lines
7.7 KiB
C#
Raw Normal View History

2026-04-25 18:20:16 +07:00
using UnityEngine;
using UnityEngine.UIElements;
2026-04-26 04:39:59 +07:00
using System.Collections;
using System.Collections.Generic;
2026-04-25 18:20:16 +07:00
namespace UI
{
public class MainMenuController : MonoBehaviour
{
2026-04-27 13:27:40 +07:00
private VisualElement _root;
2026-04-26 04:39:59 +07:00
private VisualElement _logoContainer;
private VisualElement _logo;
private VisualElement _ribbon;
2026-04-26 05:02:49 +07:00
private VisualElement _logoPlaceholder;
2026-04-26 04:39:59 +07:00
private bool _isActive = false;
2026-04-26 05:20:47 +07:00
[Header("Animation Settings")]
2026-04-26 04:39:59 +07:00
public float transitionDuration = 0.5f;
2026-04-26 05:20:47 +07:00
public float idleTimeout = 5f;
public float pulseSpeed = 2f;
public float pulseAmount = 0.05f;
private float _lastInteractionTime;
2026-04-27 13:27:40 +07:00
private int _lastClickFrame = -1;
private Coroutine _currentTransition;
2026-04-25 18:20:16 +07:00
private void OnEnable()
{
2026-04-27 13:27:40 +07:00
_root = GetComponent<UIDocument>().rootVisualElement;
2026-04-26 04:39:59 +07:00
2026-04-27 13:27:40 +07:00
_logoContainer = _root.Q<VisualElement>("beat-logo-container");
_logo = _root.Q<VisualElement>("beat-logo");
_ribbon = _root.Q<VisualElement>("menu-ribbon");
_logoPlaceholder = _root.Q<VisualElement>("logo-placeholder");
2026-04-26 04:39:59 +07:00
2026-04-27 15:48:17 +07:00
// Đảm bảo Logo luôn có thể nhấn được
_logoContainer.pickingMode = PickingMode.Position;
_logo.pickingMode = PickingMode.Position;
2026-04-26 04:39:59 +07:00
_logoContainer.RegisterCallback<ClickEvent>(OnLogoClicked);
2026-04-25 18:20:16 +07:00
2026-04-27 13:27:40 +07:00
_root.RegisterCallback<MouseMoveEvent>(evt => ResetIdleTimer());
2026-04-27 15:48:17 +07:00
2026-04-27 13:27:40 +07:00
var buttons = _root.Query<Button>().ToList();
2026-04-27 15:48:17 +07:00
foreach (var btn in buttons)
{
btn.RegisterCallback<PointerDownEvent>(evt => ApplyDrumHit(btn, true));
btn.RegisterCallback<PointerUpEvent>(evt => ApplyDrumHit(btn, false));
btn.RegisterCallback<ClickEvent>(evt => ResetIdleTimer());
}
_logoContainer.RegisterCallback<PointerDownEvent>(evt => ApplyDrumHit(_logoContainer, true));
_logoContainer.RegisterCallback<PointerUpEvent>(evt => ApplyDrumHit(_logoContainer, false));
2026-04-26 05:20:47 +07:00
2026-04-26 05:02:49 +07:00
// Routing
2026-04-27 13:27:40 +07:00
_root.Q<Button>("btn-create")?.RegisterCallback<ClickEvent>(ev => NavigateToLobby(true));
_root.Q<Button>("btn-join")?.RegisterCallback<ClickEvent>(ev => NavigateToLobby(false));
_root.Q<Button>("btn-settings")?.RegisterCallback<ClickEvent>(ev => UIManager.Instance.ToggleSettings());
_root.Q<Button>("btn-profile")?.RegisterCallback<ClickEvent>(ev => UIManager.Instance.ShowScreen("Profile"));
_root.Q<Button>("btn-exit")?.RegisterCallback<ClickEvent>(ev => Application.Quit());
2026-04-26 05:20:47 +07:00
ResetToIdleState();
}
private void Update()
{
2026-04-27 15:48:17 +07:00
float baseScale = _isActive ? 0.35f : 1.0f;
2026-04-26 05:20:47 +07:00
float pulse = Mathf.Sin(Time.time * pulseSpeed) * pulseAmount;
_logo.style.scale = new Scale(new Vector3(baseScale + pulse, baseScale + pulse, 1f));
2026-04-27 13:27:40 +07:00
if (_isActive && _currentTransition == null)
2026-04-26 05:20:47 +07:00
{
if (Time.time - _lastInteractionTime > idleTimeout)
2026-04-27 13:27:40 +07:00
_currentTransition = StartCoroutine(TransitionToIdle());
2026-04-26 05:20:47 +07:00
}
}
2026-04-26 04:39:59 +07:00
private void OnLogoClicked(ClickEvent evt)
{
2026-04-27 13:27:40 +07:00
if (Time.frameCount == _lastClickFrame) return;
_lastClickFrame = Time.frameCount;
2026-04-26 05:20:47 +07:00
ResetIdleTimer();
2026-04-27 13:27:40 +07:00
2026-04-27 15:48:17 +07:00
// QUAN TRỌNG: Chỉ vào Lobby nếu ĐÃ Active và KHÔNG đang chuyển cảnh
2026-04-26 05:02:49 +07:00
if (!_isActive) {
2026-04-27 13:27:40 +07:00
if (_currentTransition != null) StopCoroutine(_currentTransition);
_currentTransition = StartCoroutine(TransitionToActive());
2026-04-27 15:48:17 +07:00
} else if (_currentTransition == null) {
NavigateToLobby(true);
}
}
private void ApplyDrumHit(VisualElement element, bool isDown)
{
if (isDown)
{
element.style.scale = new Scale(new Vector3(0.85f, 0.85f, 1f));
element.style.transitionDuration = new List<TimeValue> { new TimeValue(0.05f, TimeUnit.Second) };
}
else
{
element.style.scale = new Scale(Vector3.one);
element.style.transitionDuration = new List<TimeValue> { new TimeValue(0.15f, TimeUnit.Second) };
2026-04-26 05:02:49 +07:00
}
2026-04-26 04:39:59 +07:00
}
2026-04-26 05:20:47 +07:00
private void ResetIdleTimer()
{
_lastInteractionTime = Time.time;
}
2026-04-27 13:27:40 +07:00
private void NavigateToLobby(bool isCreate)
{
var lobby = Object.FindFirstObjectByType<LobbyController>();
lobby?.SetMode(isCreate);
UIManager.Instance.ShowScreen("Lobby");
}
2026-04-26 05:20:47 +07:00
private void ResetToIdleState()
{
_isActive = false;
2026-04-27 13:27:40 +07:00
UIManager.Instance.isMainMenuActive = false;
2026-04-26 05:20:47 +07:00
_ribbon.style.display = DisplayStyle.None;
2026-04-27 13:27:40 +07:00
2026-04-27 15:48:17 +07:00
_root.Add(_logoContainer);
2026-04-27 13:27:40 +07:00
_logoContainer.style.position = Position.Absolute;
2026-04-27 15:48:17 +07:00
_logoContainer.style.width = 300; _logoContainer.style.height = 300;
_logoContainer.style.left = Length.Percent(50);
_logoContainer.style.top = Length.Percent(50);
_logoContainer.style.translate = new Translate(Length.Percent(-50), Length.Percent(-50));
2026-04-27 13:27:40 +07:00
_currentTransition = null;
2026-04-26 05:20:47 +07:00
}
2026-04-26 04:39:59 +07:00
private IEnumerator TransitionToActive()
{
_isActive = true;
2026-04-27 13:27:40 +07:00
UIManager.Instance.isMainMenuActive = true;
2026-04-26 05:20:47 +07:00
ResetIdleTimer();
2026-04-26 05:02:49 +07:00
_ribbon.style.display = DisplayStyle.Flex;
_ribbon.style.opacity = 0;
2026-04-27 13:27:40 +07:00
yield return null;
2026-04-26 05:20:47 +07:00
_logoContainer.style.transitionProperty = new List<StylePropertyName> { "translate", "opacity" };
2026-04-26 04:39:59 +07:00
_logoContainer.style.transitionDuration = new List<TimeValue> { new TimeValue(transitionDuration, TimeUnit.Second) };
_ribbon.style.transitionProperty = new List<StylePropertyName> { "opacity" };
_ribbon.style.transitionDuration = new List<TimeValue> { new TimeValue(transitionDuration, TimeUnit.Second) };
2026-04-26 05:20:47 +07:00
2026-04-27 15:48:17 +07:00
// Trượt Logo từ tâm sang vị trí ribbon (#2)
_logoContainer.style.translate = new Translate(Length.Percent(-75f), Length.Percent(-50f));
2026-04-26 04:39:59 +07:00
_ribbon.style.opacity = 1;
2026-04-26 05:02:49 +07:00
yield return new WaitForSeconds(transitionDuration);
2026-04-27 13:27:40 +07:00
2026-04-27 15:48:17 +07:00
// Gán chặt vào placeholder và khóa kích thước để chống giãn
2026-04-27 13:27:40 +07:00
_logoPlaceholder.Add(_logoContainer);
2026-04-27 15:48:17 +07:00
_logoContainer.style.position = Position.Relative;
_logoContainer.style.left = StyleKeyword.Auto;
_logoContainer.style.top = StyleKeyword.Auto;
2026-04-27 13:27:40 +07:00
_logoContainer.style.translate = new Translate(0, 0);
2026-04-27 15:48:17 +07:00
// Khóa kích thước nhỏ để fit vào Ribbon
_logoContainer.style.width = 100;
_logoContainer.style.height = 100;
2026-04-27 13:27:40 +07:00
_currentTransition = null;
2026-04-26 04:39:59 +07:00
}
2026-04-26 05:20:47 +07:00
private IEnumerator TransitionToIdle()
{
_isActive = false;
2026-04-27 13:27:40 +07:00
UIManager.Instance.isMainMenuActive = false;
_root.Add(_logoContainer);
_logoContainer.style.position = Position.Absolute;
2026-04-27 15:48:17 +07:00
_logoContainer.style.width = 300; _logoContainer.style.height = 300;
_logoContainer.style.left = Length.Percent(50);
_logoContainer.style.top = Length.Percent(50);
_logoContainer.style.translate = new Translate(Length.Percent(-75f), Length.Percent(-50f));
2026-04-27 13:27:40 +07:00
2026-04-27 15:48:17 +07:00
yield return null;
2026-04-26 05:20:47 +07:00
2026-04-27 15:48:17 +07:00
_logoContainer.style.translate = new Translate(Length.Percent(-50f), Length.Percent(-50f));
2026-04-26 05:20:47 +07:00
_ribbon.style.opacity = 0;
yield return new WaitForSeconds(transitionDuration);
_ribbon.style.display = DisplayStyle.None;
2026-04-27 13:27:40 +07:00
_currentTransition = null;
2026-04-26 05:20:47 +07:00
}
2026-04-25 18:20:16 +07:00
}
}