您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
103 行
3.4 KiB
103 行
3.4 KiB
using System;
|
|
using UnityEngine;
|
|
using UnityEngine.EventSystems;
|
|
using UnityEngine.InputSystem;
|
|
|
|
public class MenuSelectionHandler : MonoBehaviour
|
|
{
|
|
public GameObject _currentSelection;
|
|
public GameObject _mouseSelection;
|
|
[SerializeField] private InputReader _inputReader;
|
|
|
|
private void OnEnable()
|
|
{
|
|
_inputReader.MouseMoveMenuEvent += HandleMoveCursor;
|
|
_inputReader.MoveSelectionMenuEvent += HandleMoveSelection;
|
|
}
|
|
|
|
private void OnDisable()
|
|
{
|
|
_inputReader.MouseMoveMenuEvent -= HandleMoveCursor;
|
|
_inputReader.MoveSelectionMenuEvent -= HandleMoveSelection;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Fired by keyboard and gamepad inputs. current selected UI element will be the ui Element that was selected
|
|
/// when the event was fired. The _currentSelection is updated later on, after the EventSystem moves to the
|
|
/// desired UI element, the UI element will call into UpdateSelection()
|
|
/// </summary>
|
|
private void HandleMoveSelection()
|
|
{
|
|
Cursor.visible = false;
|
|
|
|
// Handle case where no UI element is selected because mouse left selectable bounds
|
|
if (EventSystem.current.currentSelectedGameObject == null)
|
|
EventSystem.current.SetSelectedGameObject(_currentSelection);
|
|
|
|
// occurs when mouse is on top of some button, and we hit a gamepad or keyboard key to change the selection
|
|
var mouseIsOverSelectionStart = _mouseSelection == _currentSelection;
|
|
|
|
if (mouseIsOverSelectionStart)
|
|
{
|
|
// fire pointer exit event because we don't want the button to be in the 'highlighted' state
|
|
ExecuteEvents.Execute(EventSystem.current.currentSelectedGameObject,
|
|
new PointerEventData(EventSystem.current),
|
|
ExecuteEvents.pointerExitHandler);
|
|
// update the selection
|
|
EventSystem.current.SetSelectedGameObject(_currentSelection);
|
|
|
|
// the event we fired earlier clears _mouseSelection, reset it here because our cursor is still over the button
|
|
_mouseSelection = _currentSelection;
|
|
}
|
|
}
|
|
|
|
private void HandleMoveCursor()
|
|
{
|
|
if (_mouseSelection != null)
|
|
{
|
|
EventSystem.current.SetSelectedGameObject(_mouseSelection);
|
|
}
|
|
|
|
Cursor.visible = true;
|
|
}
|
|
|
|
public void HandleMouseEnter(GameObject uiElement)
|
|
{
|
|
_mouseSelection = uiElement;
|
|
EventSystem.current.SetSelectedGameObject(uiElement);
|
|
}
|
|
|
|
public void HandleMouseExit(GameObject uiElement)
|
|
{
|
|
if (EventSystem.current.currentSelectedGameObject != uiElement) return;
|
|
|
|
// deselect UI element if mouse moves away from it
|
|
_mouseSelection = null;
|
|
EventSystem.current.SetSelectedGameObject(null);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Method interactable UI elements should call on Submit interaction to determine whether to continue or not.
|
|
/// </summary>
|
|
/// <returns></returns>
|
|
public bool AllowsSubmitOccurance()
|
|
{
|
|
// if LMB is not down, there is no edge case to handle, allow the event to continue
|
|
return !_inputReader.LeftMouseDown()
|
|
// if we know mouse & keyboard are on different elements, do not allow interaction to continue
|
|
|| _mouseSelection != null && _mouseSelection == _currentSelection;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Fired by gamepad or keyboard navigation inputs
|
|
/// </summary>
|
|
/// <param name="uiElement"></param>
|
|
public void UpdateSelection(GameObject uiElement) => _currentSelection = uiElement;
|
|
|
|
// debug
|
|
// private void OnGUI()
|
|
// {
|
|
// GUILayout.Box($"_currentSelection: {(_currentSelection != null ? _currentSelection.name : "null")}");
|
|
// GUILayout.Box($"_mouseSelection: {(_mouseSelection != null ? _mouseSelection.name : "null")}");
|
|
// }
|
|
}
|