/// Credit David Gileadi /// Sourced from - https://bitbucket.org/UnityUIExtensions/unity-ui-extensions/pull-requests/12 using System; using UnityEngine.Events; using UnityEngine.EventSystems; namespace UnityEngine.UI.Extensions { // Segmented control, like a group of buttons [AddComponentMenu("UI/Extensions/Segmented Control")] [RequireComponent(typeof(RectTransform))] public class SegmentedControl : UIBehaviour { private Selectable[] m_segments; [SerializeField] [Tooltip("A GameObject with an Image to use as a separator between segments. Size of the RectTransform will determine the size of the separator used.\nNote, make sure to disable the separator GO so that it does not affect the scene")] private Graphic m_separator; private float m_separatorWidth = 0; [SerializeField] [Tooltip("When True, it allows each button to be toggled on/off")] private bool m_allowSwitchingOff = false; [SerializeField] [Tooltip("The selected default for the control (zero indexed array)")] private int m_selectedSegmentIndex = -1; // Event delegates triggered on click. [SerializeField] [Tooltip("Event to fire once the selection has been changed")] private SegmentSelectedEvent m_onValueChanged = new SegmentSelectedEvent(); protected internal Selectable selectedSegment; protected float SeparatorWidth { get { if (m_separatorWidth == 0 && separator) { m_separatorWidth = separator.rectTransform.rect.width; var image = separator.GetComponent(); if (image) m_separatorWidth /= image.pixelsPerUnit; } return m_separatorWidth; } } [Serializable] public class SegmentSelectedEvent : UnityEvent { } public Selectable[] segments { get { if (m_segments == null || m_segments.Length == 0) { m_segments = GetChildSegments(); } return m_segments; } } [SerializeField] public Color selectedColor; public Graphic separator { get { return m_separator; } set { m_separator = value; m_separatorWidth = 0; LayoutSegments(); } } public bool allowSwitchingOff { get { return m_allowSwitchingOff; } set { m_allowSwitchingOff = value; } } public int selectedSegmentIndex { get { return Array.IndexOf(segments, selectedSegment); } set { value = Math.Max(value, -1); value = Math.Min(value, segments.Length - 1); if (value == m_selectedSegmentIndex) return; m_selectedSegmentIndex = value; if (selectedSegment) { var segment = selectedSegment.GetComponent(); if (segment) { segment.selected = false; } selectedSegment = null; } if (value != -1) { selectedSegment = segments[value]; var segment = selectedSegment.GetComponent(); if (segment) { #if UNITY_EDITOR segment.StoreTextColor(); #endif segment.selected = true; } } } } public SegmentSelectedEvent onValueChanged { get { return m_onValueChanged; } set { m_onValueChanged = value; } } protected SegmentedControl() { } protected override void Start() { base.Start(); LayoutSegments(); if (m_selectedSegmentIndex != -1) selectedSegmentIndex = m_selectedSegmentIndex; } #if UNITY_EDITOR protected override void OnValidate() { base.OnValidate(); LayoutSegments(); if (m_selectedSegmentIndex != -1) selectedSegmentIndex = m_selectedSegmentIndex; if (m_selectedSegmentIndex > transform.childCount) { selectedSegmentIndex = transform.childCount - 1; } if (selectedColor == new Color(0, 0, 0, 0)) { selectedColor = new Color(0f, 0.455f, 0.894f); } } #endif private Selectable[] GetChildSegments() { var buttons = GetComponentsInChildren(); if (buttons.Length < 2) { throw new InvalidOperationException("A segmented control must have at least two Button children"); } for (int i = 0; i < buttons.Length; i++) { var segment = buttons[i].GetComponent(); if (segment == null) { segment = buttons[i].gameObject.AddComponent(); } segment.index = i; segment.segmentedControl = this; } return buttons; } private void RecreateSprites() { for (int i = 0; i < segments.Length; i++) { if (segments[i].image == null) continue; var sprite = CutSprite(segments[i].image.sprite, i == 0, i == segments.Length - 1); var segment = segments[i].GetComponent(); if (segment) { segment.cutSprite = sprite; } segments[i].image.overrideSprite = sprite; } } static internal Sprite CutSprite(Sprite sprite, bool leftmost, bool rightmost) { if (sprite.border.x == 0 || sprite.border.z == 0) return sprite; var rect = sprite.rect; var border = sprite.border; if (!leftmost) { rect.xMin = border.x; border.x = 0; } if (!rightmost) { rect.xMax = border.z; border.z = 0; } return Sprite.Create(sprite.texture, rect, sprite.pivot, sprite.pixelsPerUnit, 0, SpriteMeshType.FullRect, border); } public void LayoutSegments() { RecreateSprites(); RectTransform transform = this.transform as RectTransform; float width = (transform.rect.width / segments.Length) - (SeparatorWidth * (segments.Length - 1)); for (int i = 0; i < segments.Length; i++) { float insetX = ((width + SeparatorWidth) * i); var rectTransform = segments[i].GetComponent(); rectTransform.anchorMin = Vector2.zero; rectTransform.anchorMax = Vector2.zero; rectTransform.SetInsetAndSizeFromParentEdge(RectTransform.Edge.Left, insetX, width); rectTransform.SetInsetAndSizeFromParentEdge(RectTransform.Edge.Top, 0, transform.rect.height); if (separator && i > 0) { var sepTransform = gameObject.transform.Find("Separator " + i); Graphic sep = (sepTransform != null) ? sepTransform.GetComponent() : (GameObject.Instantiate(separator.gameObject) as GameObject).GetComponent(); sep.gameObject.name = "Separator " + i; sep.gameObject.SetActive(true); sep.rectTransform.SetParent(this.transform, false); sep.rectTransform.anchorMin = Vector2.zero; sep.rectTransform.anchorMax = Vector2.zero; sep.rectTransform.SetInsetAndSizeFromParentEdge(RectTransform.Edge.Left, insetX - SeparatorWidth, SeparatorWidth); sep.rectTransform.SetInsetAndSizeFromParentEdge(RectTransform.Edge.Top, 0, transform.rect.height); } // TODO: maybe adjust text position } } } [RequireComponent(typeof(Selectable))] public class Segment : UIBehaviour, IPointerClickHandler, ISubmitHandler, IPointerEnterHandler, IPointerExitHandler, IPointerDownHandler, IPointerUpHandler, ISelectHandler, IDeselectHandler { internal int index; internal SegmentedControl segmentedControl; internal bool leftmost { get { return index == 0; } } internal bool rightmost { get { return index == segmentedControl.segments.Length - 1; } } public bool selected { get { return segmentedControl.selectedSegment == this.button; } set { SetSelected(value); } } internal Selectable button { get { return GetComponent(); } } [SerializeField] Color textColor; internal Sprite cutSprite; protected Segment() { } public virtual void OnPointerClick(PointerEventData eventData) { if (eventData.button != PointerEventData.InputButton.Left) return; selected = true; } public virtual void OnPointerEnter(PointerEventData eventData) { MaintainSelection(); } public virtual void OnPointerExit(PointerEventData eventData) { MaintainSelection(); } public virtual void OnPointerDown(PointerEventData eventData) { MaintainSelection(); } public virtual void OnPointerUp(PointerEventData eventData) { MaintainSelection(); } public virtual void OnSelect(BaseEventData eventData) { MaintainSelection(); } public virtual void OnDeselect(BaseEventData eventData) { MaintainSelection(); } protected override void OnEnable() { base.OnEnable(); if (segmentedControl) MaintainSelection(); } public virtual void OnSubmit(BaseEventData eventData) { selected = true; } private void SetSelected(bool value) { if (value && button.IsActive() && button.IsInteractable()) { if (segmentedControl.selectedSegment == this.button) { if (segmentedControl.allowSwitchingOff) { Deselect(); } else { MaintainSelection(); } } else { if (segmentedControl.selectedSegment) { var segment = segmentedControl.selectedSegment.GetComponent(); segmentedControl.selectedSegment = null; if (segment) { segment.TransitionButton(); } } segmentedControl.selectedSegment = this.button; StoreTextColor(); TransitionButton(); segmentedControl.onValueChanged.Invoke(index); } } else if (segmentedControl.selectedSegment == this.button) { Deselect(); } } private void Deselect() { segmentedControl.selectedSegment = null; TransitionButton(); segmentedControl.onValueChanged.Invoke(-1); } void MaintainSelection() { if (button != segmentedControl.selectedSegment) return; TransitionButton(true); } internal void TransitionButton() { TransitionButton(false); } internal void TransitionButton(bool instant) { Color tintColor = selected ? segmentedControl.selectedColor : button.colors.normalColor; Color textColor = selected ? button.colors.normalColor : this.textColor; Sprite transitionSprite = selected ? button.spriteState.pressedSprite : cutSprite; string triggerName = selected ? button.animationTriggers.pressedTrigger : button.animationTriggers.normalTrigger; switch (button.transition) { case Selectable.Transition.ColorTint: button.image.overrideSprite = cutSprite; StartColorTween(tintColor * button.colors.colorMultiplier, instant); ChangeTextColor(textColor * button.colors.colorMultiplier); break; case Selectable.Transition.SpriteSwap: if (transitionSprite != cutSprite) transitionSprite = SegmentedControl.CutSprite(transitionSprite, leftmost, rightmost); DoSpriteSwap(transitionSprite); break; case Selectable.Transition.Animation: button.image.overrideSprite = cutSprite; TriggerAnimation(triggerName); break; } } void StartColorTween(Color targetColor, bool instant) { if (button.targetGraphic == null) return; button.targetGraphic.CrossFadeColor(targetColor, instant ? 0f : button.colors.fadeDuration, true, true); } internal void StoreTextColor() { var text = GetComponentInChildren(); if (!text) return; textColor = text.color; } void ChangeTextColor(Color targetColor) { var text = GetComponentInChildren(); if (!text) return; text.color = targetColor; } void DoSpriteSwap(Sprite newSprite) { if (button.image == null) return; button.image.overrideSprite = newSprite; } void TriggerAnimation(string triggername) { if (button.animator == null || !button.animator.isActiveAndEnabled || !button.animator.hasBoundPlayables || string.IsNullOrEmpty(triggername)) return; button.animator.ResetTrigger(button.animationTriggers.normalTrigger); button.animator.ResetTrigger(button.animationTriggers.pressedTrigger); button.animator.ResetTrigger(button.animationTriggers.highlightedTrigger); button.animator.ResetTrigger(button.animationTriggers.disabledTrigger); button.animator.SetTrigger(triggername); } } }