| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759 |
- using System;
- using UnityEngine.Events;
- using UnityEngine.EventSystems;
- namespace UnityEngine.UI
- {
- [AddComponentMenu("UI/Slider", 33)]
- [ExecuteAlways]
- [RequireComponent(typeof(RectTransform))]
- /// <summary>
- /// A standard slider that can be moved between a minimum and maximum value.
- /// </summary>
- /// <remarks>
- /// The slider component is a Selectable that controls a fill, a handle, or both. The fill, when used, spans from the minimum value to the current value while the handle, when used, follow the current value.
- /// The anchors of the fill and handle RectTransforms are driven by the Slider. The fill and handle can be direct children of the GameObject with the Slider, or intermediary RectTransforms can be placed in between for additional control.
- /// When a change to the slider value occurs, a callback is sent to any registered listeners of UI.Slider.onValueChanged.
- /// </remarks>
- public class Slider : Selectable, IDragHandler, IInitializePotentialDragHandler, ICanvasElement
- {
- /// <summary>
- /// Setting that indicates one of four directions.
- /// </summary>
- public enum Direction
- {
- /// <summary>
- /// From the left to the right
- /// </summary>
- LeftToRight,
- /// <summary>
- /// From the right to the left
- /// </summary>
- RightToLeft,
- /// <summary>
- /// From the bottom to the top.
- /// </summary>
- BottomToTop,
- /// <summary>
- /// From the top to the bottom.
- /// </summary>
- TopToBottom,
- }
- [Serializable]
- /// <summary>
- /// Event type used by the UI.Slider.
- /// </summary>
- public class SliderEvent : UnityEvent<float> {}
- [SerializeField]
- private RectTransform m_FillRect;
- /// <summary>
- /// Optional RectTransform to use as fill for the slider.
- /// </summary>
- /// <example>
- /// <code>
- /// using UnityEngine;
- /// using System.Collections;
- /// using UnityEngine.UI; // Required when Using UI elements.
- ///
- /// public class Example : MonoBehaviour
- /// {
- /// public Slider mainSlider;
- /// //Reference to new "RectTransform"(Child of FillArea).
- /// public RectTransform newFillRect;
- ///
- /// //Deactivates the old FillRect and assigns a new one.
- /// void Start()
- /// {
- /// mainSlider.fillRect.gameObject.SetActive(false);
- /// mainSlider.fillRect = newFillRect;
- /// }
- /// }
- /// </code>
- /// </example>
- public RectTransform fillRect { get { return m_FillRect; } set { if (SetPropertyUtility.SetClass(ref m_FillRect, value)) {UpdateCachedReferences(); UpdateVisuals(); } } }
- [SerializeField]
- private RectTransform m_HandleRect;
- /// <summary>
- /// Optional RectTransform to use as a handle for the slider.
- /// </summary>
- /// <example>
- /// <code>
- /// using UnityEngine;
- /// using System.Collections;
- /// using UnityEngine.UI; // Required when Using UI elements.
- ///
- /// public class Example : MonoBehaviour
- /// {
- /// public Slider mainSlider;
- /// //Reference to new "RectTransform" (Child of "Handle Slide Area").
- /// public RectTransform handleHighlighted;
- ///
- /// //Deactivates the old Handle, then assigns and enables the new one.
- /// void Start()
- /// {
- /// mainSlider.handleRect.gameObject.SetActive(false);
- /// mainSlider.handleRect = handleHighlighted;
- /// mainSlider.handleRect.gameObject.SetActive(true);
- /// }
- /// }
- /// </code>
- /// </example>
- public RectTransform handleRect { get { return m_HandleRect; } set { if (SetPropertyUtility.SetClass(ref m_HandleRect, value)) { UpdateCachedReferences(); UpdateVisuals(); } } }
- [Space]
- [SerializeField]
- private Direction m_Direction = Direction.LeftToRight;
- /// <summary>
- /// The direction of the slider, from minimum to maximum value.
- /// </summary>
- /// <example>
- /// <code>
- /// using UnityEngine;
- /// using System.Collections;
- /// using UnityEngine.UI; // Required when Using UI elements.
- ///
- /// public class Example : MonoBehaviour
- /// {
- /// public Slider mainSlider;
- ///
- /// public void Start()
- /// {
- /// //Changes the direction of the slider.
- /// if (mainSlider.direction == Slider.Direction.BottomToTop)
- /// {
- /// mainSlider.direction = Slider.Direction.TopToBottom;
- /// }
- /// }
- /// }
- /// </code>
- /// </example>
- public Direction direction { get { return m_Direction; } set { if (SetPropertyUtility.SetStruct(ref m_Direction, value)) UpdateVisuals(); } }
- [SerializeField]
- private float m_MinValue = 0;
- /// <summary>
- /// The minimum allowed value of the slider.
- /// </summary>
- /// <example>
- /// <code>
- /// using UnityEngine;
- /// using System.Collections;
- /// using UnityEngine.UI; // Required when Using UI elements.
- ///
- /// public class Example : MonoBehaviour
- /// {
- /// public Slider mainSlider;
- ///
- /// void Start()
- /// {
- /// // Changes the minimum value of the slider to 10;
- /// mainSlider.minValue = 10;
- /// }
- /// }
- /// </code>
- /// </example>
- public float minValue { get { return m_MinValue; } set { if (SetPropertyUtility.SetStruct(ref m_MinValue, value)) { Set(m_Value); UpdateVisuals(); } } }
- [SerializeField]
- private float m_MaxValue = 1;
- /// <summary>
- /// The maximum allowed value of the slider.
- /// </summary>
- /// <example>
- /// <code>
- /// using UnityEngine;
- /// using System.Collections;
- /// using UnityEngine.UI; // Required when Using UI elements.
- ///
- /// public class Example : MonoBehaviour
- /// {
- /// public Slider mainSlider;
- ///
- /// void Start()
- /// {
- /// // Changes the max value of the slider to 20;
- /// mainSlider.maxValue = 20;
- /// }
- /// }
- /// </code>
- /// </example>
- public float maxValue { get { return m_MaxValue; } set { if (SetPropertyUtility.SetStruct(ref m_MaxValue, value)) { Set(m_Value); UpdateVisuals(); } } }
- [SerializeField]
- private bool m_WholeNumbers = false;
- /// <summary>
- /// Should the value only be allowed to be whole numbers?
- /// </summary>
- /// <example>
- /// <code>
- /// using UnityEngine;
- /// using System.Collections;
- /// using UnityEngine.UI; // Required when Using UI elements.
- ///
- /// public class Example : MonoBehaviour
- /// {
- /// public Slider mainSlider;
- ///
- /// public void Start()
- /// {
- /// //sets the slider's value to accept whole numbers only.
- /// mainSlider.wholeNumbers = true;
- /// }
- /// }
- /// </code>
- /// </example>
- public bool wholeNumbers { get { return m_WholeNumbers; } set { if (SetPropertyUtility.SetStruct(ref m_WholeNumbers, value)) { Set(m_Value); UpdateVisuals(); } } }
- [SerializeField]
- protected float m_Value;
- /// <summary>
- /// The current value of the slider.
- /// </summary>
- /// <example>
- /// <code>
- /// using UnityEngine;
- /// using System.Collections;
- /// using UnityEngine.UI; // Required when Using UI elements.
- ///
- /// public class Example : MonoBehaviour
- /// {
- /// public Slider mainSlider;
- ///
- /// //Invoked when a submit button is clicked.
- /// public void SubmitSliderSetting()
- /// {
- /// //Displays the value of the slider in the console.
- /// Debug.Log(mainSlider.value);
- /// }
- /// }
- /// </code>
- /// </example>
- public virtual float value
- {
- get
- {
- return wholeNumbers ? Mathf.Round(m_Value) : m_Value;
- }
- set
- {
- Set(value);
- }
- }
- /// <summary>
- /// Set the value of the slider without invoking onValueChanged callback.
- /// </summary>
- /// <param name="input">The new value for the slider.</param>
- public virtual void SetValueWithoutNotify(float input)
- {
- Set(input, false);
- }
- /// <summary>
- /// The current value of the slider normalized into a value between 0 and 1.
- /// </summary>
- /// <example>
- /// <code>
- /// using UnityEngine;
- /// using System.Collections;
- /// using UnityEngine.UI; // Required when Using UI elements.
- ///
- /// public class Example : MonoBehaviour
- /// {
- /// public Slider mainSlider;
- ///
- /// //Set to invoke when "OnValueChanged" method is called.
- /// void CheckNormalisedValue()
- /// {
- /// //Displays the normalised value of the slider everytime the value changes.
- /// Debug.Log(mainSlider.normalizedValue);
- /// }
- /// }
- /// </code>
- /// </example>
- public float normalizedValue
- {
- get
- {
- if (Mathf.Approximately(minValue, maxValue))
- return 0;
- return Mathf.InverseLerp(minValue, maxValue, value);
- }
- set
- {
- this.value = Mathf.Lerp(minValue, maxValue, value);
- }
- }
- [Space]
- [SerializeField]
- private SliderEvent m_OnValueChanged = new SliderEvent();
- /// <summary>
- /// Callback executed when the value of the slider is changed.
- /// </summary>
- /// <example>
- /// <code>
- /// using UnityEngine;
- /// using System.Collections;
- /// using UnityEngine.UI; // Required when Using UI elements.
- ///
- /// public class Example : MonoBehaviour
- /// {
- /// public Slider mainSlider;
- ///
- /// public void Start()
- /// {
- /// //Adds a listener to the main slider and invokes a method when the value changes.
- /// mainSlider.onValueChanged.AddListener(delegate {ValueChangeCheck(); });
- /// }
- ///
- /// // Invoked when the value of the slider changes.
- /// public void ValueChangeCheck()
- /// {
- /// Debug.Log(mainSlider.value);
- /// }
- /// }
- /// </code>
- /// </example>
- public SliderEvent onValueChanged { get { return m_OnValueChanged; } set { m_OnValueChanged = value; } }
- // Private fields
- private Image m_FillImage;
- private Transform m_FillTransform;
- private RectTransform m_FillContainerRect;
- private Transform m_HandleTransform;
- private RectTransform m_HandleContainerRect;
- // The offset from handle position to mouse down position
- private Vector2 m_Offset = Vector2.zero;
- // field is never assigned warning
- #pragma warning disable 649
- private DrivenRectTransformTracker m_Tracker;
- #pragma warning restore 649
- // This "delayed" mechanism is required for case 1037681.
- private bool m_DelayedUpdateVisuals = false;
- // Size of each step.
- float stepSize { get { return wholeNumbers ? 1 : (maxValue - minValue) * 0.1f; } }
- protected Slider()
- {}
- #if UNITY_EDITOR
- protected override void OnValidate()
- {
- base.OnValidate();
- if (wholeNumbers)
- {
- m_MinValue = Mathf.Round(m_MinValue);
- m_MaxValue = Mathf.Round(m_MaxValue);
- }
- //Onvalidate is called before OnEnabled. We need to make sure not to touch any other objects before OnEnable is run.
- if (IsActive())
- {
- UpdateCachedReferences();
- // Update rects in next update since other things might affect them even if value didn't change.
- m_DelayedUpdateVisuals = true;
- }
- if (!UnityEditor.PrefabUtility.IsPartOfPrefabAsset(this) && !Application.isPlaying)
- CanvasUpdateRegistry.RegisterCanvasElementForLayoutRebuild(this);
- }
- #endif // if UNITY_EDITOR
- public virtual void Rebuild(CanvasUpdate executing)
- {
- #if UNITY_EDITOR
- if (executing == CanvasUpdate.Prelayout)
- onValueChanged.Invoke(value);
- #endif
- }
- /// <summary>
- /// See ICanvasElement.LayoutComplete
- /// </summary>
- public virtual void LayoutComplete()
- {}
- /// <summary>
- /// See ICanvasElement.GraphicUpdateComplete
- /// </summary>
- public virtual void GraphicUpdateComplete()
- {}
- protected override void OnEnable()
- {
- base.OnEnable();
- UpdateCachedReferences();
- Set(m_Value, false);
- // Update rects since they need to be initialized correctly.
- UpdateVisuals();
- }
- protected override void OnDisable()
- {
- m_Tracker.Clear();
- base.OnDisable();
- }
- /// <summary>
- /// Update the rect based on the delayed update visuals.
- /// Got around issue of calling sendMessage from onValidate.
- /// </summary>
- protected virtual void Update()
- {
- if (m_DelayedUpdateVisuals)
- {
- m_DelayedUpdateVisuals = false;
- Set(m_Value, false);
- UpdateVisuals();
- }
- }
- protected override void OnDidApplyAnimationProperties()
- {
- // Has value changed? Various elements of the slider have the old normalisedValue assigned, we can use this to perform a comparison.
- // We also need to ensure the value stays within min/max.
- m_Value = ClampValue(m_Value);
- float oldNormalizedValue = normalizedValue;
- if (m_FillContainerRect != null)
- {
- if (m_FillImage != null && m_FillImage.type == Image.Type.Filled)
- oldNormalizedValue = m_FillImage.fillAmount;
- else
- oldNormalizedValue = (reverseValue ? 1 - m_FillRect.anchorMin[(int)axis] : m_FillRect.anchorMax[(int)axis]);
- }
- else if (m_HandleContainerRect != null)
- oldNormalizedValue = (reverseValue ? 1 - m_HandleRect.anchorMin[(int)axis] : m_HandleRect.anchorMin[(int)axis]);
- UpdateVisuals();
- if (oldNormalizedValue != normalizedValue)
- {
- UISystemProfilerApi.AddMarker("Slider.value", this);
- onValueChanged.Invoke(m_Value);
- }
- }
- void UpdateCachedReferences()
- {
- if (m_FillRect && m_FillRect != (RectTransform)transform)
- {
- m_FillTransform = m_FillRect.transform;
- m_FillImage = m_FillRect.GetComponent<Image>();
- if (m_FillTransform.parent != null)
- m_FillContainerRect = m_FillTransform.parent.GetComponent<RectTransform>();
- }
- else
- {
- m_FillRect = null;
- m_FillContainerRect = null;
- m_FillImage = null;
- }
- if (m_HandleRect && m_HandleRect != (RectTransform)transform)
- {
- m_HandleTransform = m_HandleRect.transform;
- if (m_HandleTransform.parent != null)
- m_HandleContainerRect = m_HandleTransform.parent.GetComponent<RectTransform>();
- }
- else
- {
- m_HandleRect = null;
- m_HandleContainerRect = null;
- }
- }
- float ClampValue(float input)
- {
- float newValue = Mathf.Clamp(input, minValue, maxValue);
- if (wholeNumbers)
- newValue = Mathf.Round(newValue);
- return newValue;
- }
- /// <summary>
- /// Set the value of the slider.
- /// </summary>
- /// <param name="input">The new value for the slider.</param>
- /// <param name="sendCallback">If the OnValueChanged callback should be invoked.</param>
- /// <remarks>
- /// Process the input to ensure the value is between min and max value. If the input is different set the value and send the callback is required.
- /// </remarks>
- protected virtual void Set(float input, bool sendCallback = true)
- {
- // Clamp the input
- float newValue = ClampValue(input);
- // If the stepped value doesn't match the last one, it's time to update
- if (m_Value == newValue)
- return;
- m_Value = newValue;
- UpdateVisuals();
- if (sendCallback)
- {
- UISystemProfilerApi.AddMarker("Slider.value", this);
- m_OnValueChanged.Invoke(newValue);
- }
- }
- protected override void OnRectTransformDimensionsChange()
- {
- base.OnRectTransformDimensionsChange();
- //This can be invoked before OnEnabled is called. So we shouldn't be accessing other objects, before OnEnable is called.
- if (!IsActive())
- return;
- UpdateVisuals();
- }
- enum Axis
- {
- Horizontal = 0,
- Vertical = 1
- }
- Axis axis { get { return (m_Direction == Direction.LeftToRight || m_Direction == Direction.RightToLeft) ? Axis.Horizontal : Axis.Vertical; } }
- bool reverseValue { get { return m_Direction == Direction.RightToLeft || m_Direction == Direction.TopToBottom; } }
- // Force-update the slider. Useful if you've changed the properties and want it to update visually.
- private void UpdateVisuals()
- {
- #if UNITY_EDITOR
- if (!Application.isPlaying)
- UpdateCachedReferences();
- #endif
- m_Tracker.Clear();
- if (m_FillContainerRect != null)
- {
- m_Tracker.Add(this, m_FillRect, DrivenTransformProperties.Anchors);
- Vector2 anchorMin = Vector2.zero;
- Vector2 anchorMax = Vector2.one;
- if (m_FillImage != null && m_FillImage.type == Image.Type.Filled)
- {
- m_FillImage.fillAmount = normalizedValue;
- }
- else
- {
- if (reverseValue)
- anchorMin[(int)axis] = 1 - normalizedValue;
- else
- anchorMax[(int)axis] = normalizedValue;
- }
- m_FillRect.anchorMin = anchorMin;
- m_FillRect.anchorMax = anchorMax;
- }
- if (m_HandleContainerRect != null)
- {
- m_Tracker.Add(this, m_HandleRect, DrivenTransformProperties.Anchors);
- Vector2 anchorMin = Vector2.zero;
- Vector2 anchorMax = Vector2.one;
- anchorMin[(int)axis] = anchorMax[(int)axis] = (reverseValue ? (1 - normalizedValue) : normalizedValue);
- m_HandleRect.anchorMin = anchorMin;
- m_HandleRect.anchorMax = anchorMax;
- }
- }
- // Update the slider's position based on the mouse.
- void UpdateDrag(PointerEventData eventData, Camera cam)
- {
- RectTransform clickRect = m_HandleContainerRect ?? m_FillContainerRect;
- if (clickRect != null && clickRect.rect.size[(int)axis] > 0)
- {
- Vector2 position = Vector2.zero;
- if (!MultipleDisplayUtilities.GetRelativeMousePositionForDrag(eventData, ref position))
- return;
- Vector2 localCursor;
- if (!RectTransformUtility.ScreenPointToLocalPointInRectangle(clickRect, position, cam, out localCursor))
- return;
- localCursor -= clickRect.rect.position;
- float val = Mathf.Clamp01((localCursor - m_Offset)[(int)axis] / clickRect.rect.size[(int)axis]);
- normalizedValue = (reverseValue ? 1f - val : val);
- }
- }
- private bool MayDrag(PointerEventData eventData)
- {
- return IsActive() && IsInteractable() && eventData.button == PointerEventData.InputButton.Left;
- }
- public override void OnPointerDown(PointerEventData eventData)
- {
- if (!MayDrag(eventData))
- return;
- base.OnPointerDown(eventData);
- m_Offset = Vector2.zero;
- if (m_HandleContainerRect != null && RectTransformUtility.RectangleContainsScreenPoint(m_HandleRect, eventData.pointerPressRaycast.screenPosition, eventData.enterEventCamera))
- {
- Vector2 localMousePos;
- if (RectTransformUtility.ScreenPointToLocalPointInRectangle(m_HandleRect, eventData.pointerPressRaycast.screenPosition, eventData.pressEventCamera, out localMousePos))
- m_Offset = localMousePos;
- }
- else
- {
- // Outside the slider handle - jump to this point instead
- UpdateDrag(eventData, eventData.pressEventCamera);
- }
- }
- public virtual void OnDrag(PointerEventData eventData)
- {
- if (!MayDrag(eventData))
- return;
- UpdateDrag(eventData, eventData.pressEventCamera);
- }
- public override void OnMove(AxisEventData eventData)
- {
- if (!IsActive() || !IsInteractable())
- {
- base.OnMove(eventData);
- return;
- }
- switch (eventData.moveDir)
- {
- case MoveDirection.Left:
- if (axis == Axis.Horizontal && FindSelectableOnLeft() == null)
- Set(reverseValue ? value + stepSize : value - stepSize);
- else
- base.OnMove(eventData);
- break;
- case MoveDirection.Right:
- if (axis == Axis.Horizontal && FindSelectableOnRight() == null)
- Set(reverseValue ? value - stepSize : value + stepSize);
- else
- base.OnMove(eventData);
- break;
- case MoveDirection.Up:
- if (axis == Axis.Vertical && FindSelectableOnUp() == null)
- Set(reverseValue ? value - stepSize : value + stepSize);
- else
- base.OnMove(eventData);
- break;
- case MoveDirection.Down:
- if (axis == Axis.Vertical && FindSelectableOnDown() == null)
- Set(reverseValue ? value + stepSize : value - stepSize);
- else
- base.OnMove(eventData);
- break;
- }
- }
- /// <summary>
- /// See Selectable.FindSelectableOnLeft
- /// </summary>
- public override Selectable FindSelectableOnLeft()
- {
- if (navigation.mode == Navigation.Mode.Automatic && axis == Axis.Horizontal)
- return null;
- return base.FindSelectableOnLeft();
- }
- /// <summary>
- /// See Selectable.FindSelectableOnRight
- /// </summary>
- public override Selectable FindSelectableOnRight()
- {
- if (navigation.mode == Navigation.Mode.Automatic && axis == Axis.Horizontal)
- return null;
- return base.FindSelectableOnRight();
- }
- /// <summary>
- /// See Selectable.FindSelectableOnUp
- /// </summary>
- public override Selectable FindSelectableOnUp()
- {
- if (navigation.mode == Navigation.Mode.Automatic && axis == Axis.Vertical)
- return null;
- return base.FindSelectableOnUp();
- }
- /// <summary>
- /// See Selectable.FindSelectableOnDown
- /// </summary>
- public override Selectable FindSelectableOnDown()
- {
- if (navigation.mode == Navigation.Mode.Automatic && axis == Axis.Vertical)
- return null;
- return base.FindSelectableOnDown();
- }
- public virtual void OnInitializePotentialDrag(PointerEventData eventData)
- {
- eventData.useDragThreshold = false;
- }
- /// <summary>
- /// Sets the direction of this slider, optionally changing the layout as well.
- /// </summary>
- /// <param name="direction">The direction of the slider</param>
- /// <param name="includeRectLayouts">Should the layout be flipped together with the slider direction</param>
- /// <example>
- /// <code>
- /// using UnityEngine;
- /// using System.Collections;
- /// using UnityEngine.UI; // Required when Using UI elements.
- ///
- /// public class Example : MonoBehaviour
- /// {
- /// public Slider mainSlider;
- ///
- /// public void Start()
- /// {
- /// mainSlider.SetDirection(Slider.Direction.LeftToRight, false);
- /// }
- /// }
- /// </code>
- /// </example>
- public void SetDirection(Direction direction, bool includeRectLayouts)
- {
- Axis oldAxis = axis;
- bool oldReverse = reverseValue;
- this.direction = direction;
- if (!includeRectLayouts)
- return;
- if (axis != oldAxis)
- RectTransformUtility.FlipLayoutAxes(transform as RectTransform, true, true);
- if (reverseValue != oldReverse)
- RectTransformUtility.FlipLayoutOnAxis(transform as RectTransform, (int)axis, true, true);
- }
- }
- }
|