276 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			276 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| /// Credit Ahmad S. Al-Faqeeh
 | |
| /// Sourced from - https://github.com/Unity-UI-Extensions/com.unity.uiextensions/issues/205
 | |
| /// Based on the UIVerticalScroller
 | |
| /// 
 | |
| using UnityEngine.Events;
 | |
| 
 | |
| namespace UnityEngine.UI.Extensions
 | |
| {
 | |
|     [RequireComponent(typeof(ScrollRect))]
 | |
|     [AddComponentMenu("Layout/Extensions/Horizontal Scroller")]
 | |
|     public class UIHorizontalScroller : MonoBehaviour
 | |
|     {
 | |
|         private float[] distReposition;
 | |
|         private float[] distance;
 | |
| 
 | |
|         [SerializeField]
 | |
|         [Tooltip("desired ScrollRect")]
 | |
|         private ScrollRect scrollRect;
 | |
| 
 | |
|         [SerializeField]
 | |
|         [Tooltip("Elements to populate inside the scroller")]
 | |
|         private GameObject[] arrayOfElements;
 | |
| 
 | |
|         [SerializeField]
 | |
|         [Tooltip("Center display area (position of zoomed content)")]
 | |
|         private RectTransform center;
 | |
| 
 | |
|         [SerializeField]
 | |
|         [Tooltip("Size / spacing of elements")]
 | |
|         private RectTransform elementSize;
 | |
| 
 | |
|         [SerializeField]
 | |
|         [Tooltip("Scale = 1/ (1+distance from center * shrinkage)")]
 | |
|         private Vector2 elementShrinkage = new Vector2(1f / 200, 1f / 200);
 | |
| 
 | |
|         [SerializeField]
 | |
|         [Tooltip("Minimum element scale (furthest from center)")]
 | |
|         private Vector2 minScale = new Vector2(0.7f, 0.7f);
 | |
| 
 | |
|         [SerializeField]
 | |
|         [Tooltip("Select the item to be in center on start. (optional)")]
 | |
|         private int startingIndex = -1;
 | |
| 
 | |
|         [SerializeField]
 | |
|         [Tooltip("Stop scrolling past last element from inertia.")]
 | |
|         private bool stopMomentumOnEnd = true;
 | |
| 
 | |
|         [SerializeField]
 | |
|         [Tooltip("Set Items out of center to not interactible.")]
 | |
|         private bool disableUnfocused = true;
 | |
| 
 | |
|         [SerializeField]
 | |
|         [Tooltip("Button to go to the next page. (optional)")]
 | |
|         private GameObject scrollLeftButton;
 | |
| 
 | |
|         [SerializeField]
 | |
|         [Tooltip("Button to go to the previous page. (optional)")]
 | |
|         private GameObject scrollRightButton;
 | |
| 
 | |
|         [SerializeField]
 | |
|         [Tooltip("Event fired when a specific item is clicked, exposes index number of item. (optional)")]
 | |
|         private UnityEvent<int> onButtonClicked;
 | |
| 
 | |
|         [SerializeField]
 | |
|         [Tooltip("Event fired when the focused item is Changed. (optional)")]
 | |
|         private UnityEvent<int> onFocusChanged;
 | |
| 
 | |
|         public int FocusedElementIndex { get; private set; }
 | |
| 
 | |
|         public RectTransform Center { get => center; set => center = value; }
 | |
| 
 | |
|         //Scrollable area (content of desired ScrollRect)
 | |
|         public RectTransform ScrollingPanel { get { return scrollRect.content; } }
 | |
| 
 | |
|         public string Result { get; private set; }
 | |
| 
 | |
|         public UIHorizontalScroller() { }
 | |
| 
 | |
|         public UIHorizontalScroller(RectTransform center, RectTransform elementSize, ScrollRect scrollRect, GameObject[] arrayOfElements)
 | |
|         {
 | |
|             this.scrollRect = scrollRect;
 | |
|             this.elementSize = elementSize;
 | |
|             this.arrayOfElements = arrayOfElements;
 | |
|             this.center = center;
 | |
|         }
 | |
| 
 | |
|         public void Awake()
 | |
|         {
 | |
|             if (!scrollRect)
 | |
|             {
 | |
|                 scrollRect = GetComponent<ScrollRect>();
 | |
|             }
 | |
| 
 | |
|             if (!center)
 | |
|             {
 | |
|                 Debug.LogError("Please define the RectTransform for the Center viewport of the scrollable area");
 | |
|             }
 | |
| 
 | |
|             if (!elementSize)
 | |
|             {
 | |
|                 elementSize = center;
 | |
|             }
 | |
| 
 | |
|             if (arrayOfElements == null || arrayOfElements.Length == 0)
 | |
|             {
 | |
|                 var childCount = scrollRect.content.childCount;
 | |
|                 if (childCount > 0)
 | |
|                 {
 | |
|                     arrayOfElements = new GameObject[childCount];
 | |
|                     for (int i = 0; i < childCount; i++)
 | |
|                     {
 | |
|                         arrayOfElements[i] = scrollRect.content.GetChild(i).gameObject;
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public void Start()
 | |
|         {
 | |
|             if (scrollLeftButton)
 | |
|             {
 | |
|                 scrollLeftButton.GetComponent<Button>().onClick.AddListener(() => ScrollLeft());
 | |
|             }
 | |
| 
 | |
|             if (scrollRightButton)
 | |
|             {
 | |
|                 scrollRightButton.GetComponent<Button>().onClick.AddListener(() => ScrollRight());
 | |
|             }
 | |
| 
 | |
|             UpdateChildren(startingIndex, arrayOfElements);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Recognises and resizes the children.
 | |
|         /// </summary>
 | |
|         /// <param name="startingIndex">Starting index.</param>
 | |
|         /// <param name="arrayOfElements">Array of elements.</param>
 | |
|         public void UpdateChildren(int startingIndex = -1, GameObject[] arrayOfElements = null)
 | |
|         {
 | |
|             // Set _arrayOfElements to arrayOfElements if given, otherwise to child objects of the scrolling panel.
 | |
|             if (arrayOfElements != null)
 | |
|             {
 | |
|                 this.arrayOfElements = arrayOfElements;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 this.arrayOfElements = new GameObject[ScrollingPanel.childCount];
 | |
|                 for (int i = 0; i < ScrollingPanel.childCount; i++)
 | |
|                 {
 | |
|                     this.arrayOfElements[i] = ScrollingPanel.GetChild(i).gameObject;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             // resize the elements to match elementSize rect
 | |
|             for (var i = 0; i < this.arrayOfElements.Length; i++)
 | |
|             {
 | |
|                 AddListener(arrayOfElements[i], i);
 | |
| 
 | |
|                 RectTransform r = this.arrayOfElements[i].GetComponent<RectTransform>();
 | |
|                 r.anchorMax = r.anchorMin = r.pivot = new Vector2(0.5f, 0.5f);
 | |
|                 r.localPosition = new Vector2(i * elementSize.rect.size.x,0);
 | |
|                 r.sizeDelta = elementSize.rect.size;
 | |
|             }
 | |
| 
 | |
|             // prepare for scrolling
 | |
|             distance = new float[this.arrayOfElements.Length];
 | |
|             distReposition = new float[this.arrayOfElements.Length];
 | |
|             FocusedElementIndex = -1;
 | |
| 
 | |
|             // if starting index is given, snap to respective element
 | |
|             if (startingIndex > -1)
 | |
|             {
 | |
|                 startingIndex = startingIndex > this.arrayOfElements.Length ? this.arrayOfElements.Length - 1 : startingIndex;
 | |
|                 SnapToElement(startingIndex);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private void AddListener(GameObject button, int index)
 | |
|         {
 | |
|             var buttonClick = button.GetComponent<Button>();
 | |
|             buttonClick.onClick.RemoveAllListeners();
 | |
|             buttonClick.onClick.AddListener(() => onButtonClicked?.Invoke(index));
 | |
|         }
 | |
| 
 | |
|         public void Update()
 | |
|         {
 | |
|             if (arrayOfElements.Length < 1)
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             for (var i = 0; i < arrayOfElements.Length; i++)
 | |
|             {
 | |
|                 var arrayElementRT = arrayOfElements[i].GetComponent<RectTransform>();
 | |
|                 distReposition[i] = center.position.x - arrayElementRT.position.x;
 | |
|                 distance[i] = Mathf.Abs(distReposition[i]);
 | |
| 
 | |
|                 //Magnifying effect
 | |
|                 Vector2 scale = Vector2.Max(minScale, new Vector2(1 / (1 + distance[i] * elementShrinkage.x), (1 / (1 + distance[i] * elementShrinkage.y))));
 | |
|                 arrayElementRT.transform.localScale = new Vector3(scale.x, scale.y, 1f);
 | |
|             }
 | |
| 
 | |
|             float minDistance = Mathf.Min(distance);
 | |
|             int oldFocusedElement = FocusedElementIndex;
 | |
| 
 | |
|             for (var i = 0; i < arrayOfElements.Length; i++)
 | |
|             {
 | |
|                 arrayOfElements[i].GetComponent<CanvasGroup>().interactable = !disableUnfocused || minDistance == distance[i];
 | |
|                 if (minDistance == distance[i])
 | |
|                 {
 | |
|                     FocusedElementIndex = i;
 | |
| #if UNITY_2022_1_OR_NEWER
 | |
|                     var textComponentTxtMeshPro = arrayOfElements[i].GetComponentInChildren<TMPro.TMP_Text>();
 | |
|                     if (textComponentTxtMeshPro != null)
 | |
|                     {
 | |
|                         Result = textComponentTxtMeshPro.text;
 | |
|                     }
 | |
| #else
 | |
|                     var textComponent = arrayOfElements[i].GetComponentInChildren<Text>();
 | |
|                     if (textComponent != null)
 | |
|                     {
 | |
|                         Result = textComponent.text;
 | |
|                     }
 | |
| #endif
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (FocusedElementIndex != oldFocusedElement && onFocusChanged != null)
 | |
|             {
 | |
|                 onFocusChanged.Invoke(FocusedElementIndex);
 | |
|             }
 | |
| 
 | |
|             if (!UIExtensionsInputManager.GetMouseButton(0))
 | |
|             {
 | |
|                 // scroll slowly to nearest element when not dragged
 | |
|                 ScrollingElements();
 | |
|             }
 | |
| 
 | |
|             // stop scrolling past last element from inertia
 | |
|             if (stopMomentumOnEnd
 | |
|                 && (arrayOfElements[0].GetComponent<RectTransform>().position.x > center.position.x
 | |
|                 || arrayOfElements[arrayOfElements.Length - 1].GetComponent<RectTransform>().position.x < center.position.x))
 | |
|             {
 | |
|                 scrollRect.velocity = Vector2.zero;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private void ScrollingElements()
 | |
|         {
 | |
|             float newX = Mathf.Lerp(ScrollingPanel.anchoredPosition.x, ScrollingPanel.anchoredPosition.x + distReposition[FocusedElementIndex], Time.deltaTime * 2f);
 | |
|             Vector2 newPosition = new Vector2(newX, ScrollingPanel.anchoredPosition.y);
 | |
|             ScrollingPanel.anchoredPosition = newPosition;
 | |
|         }
 | |
| 
 | |
|         public void SnapToElement(int element)
 | |
|         {
 | |
|             float deltaElementPositionX = elementSize.rect.width / 1.2f * element;
 | |
|             Vector2 newPosition = new Vector2(-deltaElementPositionX, ScrollingPanel.anchoredPosition.y);
 | |
|             ScrollingPanel.anchoredPosition = newPosition;
 | |
|         }
 | |
| 
 | |
|         public void ScrollLeft()
 | |
|         {
 | |
|             float deltaLeft = elementSize.rect.width / 1.2f;
 | |
|             Vector2 newPositionLeft = new Vector2(ScrollingPanel.anchoredPosition.x - deltaLeft, ScrollingPanel.anchoredPosition.y);
 | |
|             ScrollingPanel.anchoredPosition = Vector2.Lerp(ScrollingPanel.anchoredPosition, newPositionLeft, 1);
 | |
|         }
 | |
| 
 | |
|         public void ScrollRight()
 | |
|         {
 | |
|             float deltaRight = elementSize.rect.width / 1.2f;// arrayOfElements[0].GetComponent<RectTransform>().rect.width;
 | |
|             Vector2 newPositionRight = new Vector2(ScrollingPanel.anchoredPosition.x + deltaRight, ScrollingPanel.anchoredPosition.y);
 | |
|             ScrollingPanel.anchoredPosition = newPositionRight;
 | |
|         }
 | |
|     }
 | |
| } | 
