diff --git a/Editor/UIExtensionsMenuOptions.cs b/Editor/UIExtensionsMenuOptions.cs index 6f2a64c..cfce769 100644 --- a/Editor/UIExtensionsMenuOptions.cs +++ b/Editor/UIExtensionsMenuOptions.cs @@ -708,33 +708,78 @@ namespace UnityEditor.UI go.AddComponent(); Selection.activeGameObject = go; } - #endregion - - #region Accordion - [MenuItem("GameObject/UI/Extensions/Accordion/Accordion Group", false)] - static public void AddAccordionGroup(MenuCommand menuCommand) - { - GameObject go = CreateUIElementRoot("Accordion Group", menuCommand, s_ThickGUIElementSize); - go.AddComponent(); - go.AddComponent(); - go.AddComponent(); - go.AddComponent(); - Selection.activeGameObject = go; - } - - [MenuItem("GameObject/UI/Extensions/Accordion/Accordion Element", false)] - static public void AddAccordionElement(MenuCommand menuCommand) - { - GameObject go = CreateUIElementRoot("Accordion Element", menuCommand, s_ThickGUIElementSize); - go.AddComponent(); - go.AddComponent(); - Selection.activeGameObject = go; - - } #endregion - #region Drop Down controls - [MenuItem("GameObject/UI/Extensions/AutoComplete ComboBox", false)] + #region Accordion + [MenuItem("GameObject/UI/Extensions/Accordion/Accordion", false)] + static public void AddAccordionVertical(MenuCommand menuCommand) + { + GameObject go = CreateUIElementRoot("Accordion Group", menuCommand, s_ThickGUIElementSize); + CreateAccordionGroup(go); + for (int i = 0; i < 3; i++) + { + GameObject child = CreateUIObject($"Accordion Element {i}", go); + CreateAccordionElement(child); + } + Selection.activeGameObject = go; + } + + [MenuItem("GameObject/UI/Extensions/Accordion/Accordion Group", false)] + static public void AddAccordionGroup(MenuCommand menuCommand) + { + GameObject go = CreateUIElementRoot("Accordion Group", menuCommand, s_ThickGUIElementSize); + CreateAccordionGroup(go); + Selection.activeGameObject = go; + } + + private static void CreateAccordionGroup(GameObject go) + { + var vlg = go.AddComponent(); + vlg.childControlHeight = true; + vlg.childControlWidth = true; + vlg.childForceExpandHeight = false; + vlg.childForceExpandHeight = true; + var csf = go.AddComponent(); + csf.verticalFit = ContentSizeFitter.FitMode.PreferredSize; + go.AddComponent(); + go.AddComponent(); + } + + [MenuItem("GameObject/UI/Extensions/Accordion/Accordion Element", false)] + static public void AddAccordionElement(MenuCommand menuCommand) + { + GameObject go = CreateUIElementRoot("Accordion Element", menuCommand, s_ThickGUIElementSize); + CreateAccordionElement(go); + + Selection.activeGameObject = go; + } + + private static void CreateAccordionElement(GameObject go) + { + var vlg = go.AddComponent(); + vlg.childControlHeight = true; + vlg.childControlWidth = true; + vlg.childForceExpandHeight = false; + vlg.childForceExpandHeight = true; + go.AddComponent(); + var accordionElement = go.AddComponent(); + + // Header + GameObject headergo = CreateUIObject("Header", go); + var headerLayout = headergo.AddComponent(); + headerLayout.minHeight = accordionElement.MinHeight; + var headerText = headergo.AddComponent(); + headerText.text = "This is an Accordion header"; + + // Text + GameObject textgo = CreateUIObject("Text", go); + var textText = textgo.AddComponent(); + textText.text = "This is an example of text in an accordion element\nLots of information can be put here for selection\nIf you like"; + } + #endregion + + #region Drop Down controls + [MenuItem("GameObject/UI/Extensions/AutoComplete ComboBox", false)] static public void AddAutoCompleteComboBox(MenuCommand menuCommand) { GameObject autoCompleteComboBoxRoot = CreateUIElementRoot("AutoCompleteComboBox", menuCommand, s_ThickGUIElementSize); diff --git a/Examples b/Examples index 8f7a9de..cf1a370 160000 --- a/Examples +++ b/Examples @@ -1 +1 @@ -Subproject commit 8f7a9de774bcf64109a5e1362650e9f37572588f +Subproject commit cf1a370339e25bc8fcdb10b80d0d9616f12ead0c diff --git a/Runtime/Scripts/Controls/Accordion/Accordion.cs b/Runtime/Scripts/Controls/Accordion/Accordion.cs index 88b43f7..88e7074 100644 --- a/Runtime/Scripts/Controls/Accordion/Accordion.cs +++ b/Runtime/Scripts/Controls/Accordion/Accordion.cs @@ -4,11 +4,14 @@ namespace UnityEngine.UI.Extensions { - [RequireComponent(typeof(VerticalLayoutGroup), typeof(ContentSizeFitter), typeof(ToggleGroup))] + [RequireComponent(typeof(HorizontalOrVerticalLayoutGroup), typeof(ContentSizeFitter), typeof(ToggleGroup))] [AddComponentMenu("UI/Extensions/Accordion/Accordion Group")] public class Accordion : MonoBehaviour { - + private bool m_expandVertical = true; + [HideInInspector] + public bool ExpandVerticval => m_expandVertical; + public enum Transition { Instant, @@ -37,5 +40,22 @@ namespace UnityEngine.UI.Extensions get { return this.m_TransitionDuration; } set { this.m_TransitionDuration = value; } } + + private void Awake() + { + m_expandVertical = GetComponent() ? false : true; + var group = GetComponent(); + } + +#if UNITY_EDITOR + + private void OnValidate() + { + if (!GetComponent() && !GetComponent()) + { + Debug.LogError("Accordion requires either a Horizontal or Vertical Layout group to place children"); + } + } +#endif } } \ No newline at end of file diff --git a/Runtime/Scripts/Controls/Accordion/AccordionElement.cs b/Runtime/Scripts/Controls/Accordion/AccordionElement.cs index 64f20e2..1ed11db 100644 --- a/Runtime/Scripts/Controls/Accordion/AccordionElement.cs +++ b/Runtime/Scripts/Controls/Accordion/AccordionElement.cs @@ -12,7 +12,13 @@ namespace UnityEngine.UI.Extensions { [SerializeField] private float m_MinHeight = 18f; - + + public float MinHeight => m_MinHeight; + + [SerializeField] private float m_MinWidth = 40f; + + public float MinWidth => m_MinWidth; + private Accordion m_Accordion; private RectTransform m_RectTransform; private LayoutElement m_LayoutElement; @@ -43,7 +49,8 @@ namespace UnityEngine.UI.Extensions protected override void OnValidate() { base.OnValidate(); - + this.m_Accordion = this.gameObject.GetComponentInParent(); + if (this.group == null) { ToggleGroup tg = this.GetComponentInParent(); @@ -60,11 +67,26 @@ namespace UnityEngine.UI.Extensions { if (this.isOn) { - le.preferredHeight = -1f; + if (m_Accordion.ExpandVerticval) + { + le.preferredHeight = -1f; + } + else + { + le.preferredWidth = -1f; + } } else { - le.preferredHeight = this.m_MinHeight; + if (m_Accordion.ExpandVerticval) + { + le.preferredHeight = this.m_MinHeight; + } + else + { + le.preferredWidth = this.m_MinWidth; + + } } } } @@ -81,22 +103,50 @@ namespace UnityEngine.UI.Extensions { if (state) { - this.m_LayoutElement.preferredHeight = -1f; + if (m_Accordion.ExpandVerticval) + { + this.m_LayoutElement.preferredHeight = -1f; + } + else + { + this.m_LayoutElement.preferredWidth = -1f; + } } else { - this.m_LayoutElement.preferredHeight = this.m_MinHeight; + if (m_Accordion.ExpandVerticval) + { + this.m_LayoutElement.preferredHeight = this.m_MinHeight; + } + else + { + this.m_LayoutElement.preferredWidth = this.m_MinWidth; + } } } else if (transition == Accordion.Transition.Tween) { if (state) { - this.StartTween(this.m_MinHeight, this.GetExpandedHeight()); + if (m_Accordion.ExpandVerticval) + { + this.StartTween(this.m_MinHeight, this.GetExpandedHeight()); + } + else + { + this.StartTween(this.m_MinWidth, this.GetExpandedWidth()); + } } else { - this.StartTween(this.m_RectTransform.rect.height, this.m_MinHeight); + if (m_Accordion.ExpandVerticval) + { + this.StartTween(this.m_RectTransform.rect.height, this.m_MinHeight); + } + else + { + this.StartTween(this.m_RectTransform.rect.width, this.m_MinWidth); + } } } } @@ -113,7 +163,20 @@ namespace UnityEngine.UI.Extensions return h; } - + + protected float GetExpandedWidth() + { + if (this.m_LayoutElement == null) + return this.m_MinWidth; + + float originalPrefW = this.m_LayoutElement.preferredWidth; + this.m_LayoutElement.preferredWidth = -1f; + float w = LayoutUtility.GetPreferredWidth(this.m_RectTransform); + this.m_LayoutElement.preferredWidth = originalPrefW; + + return w; + } + protected void StartTween(float startFloat, float targetFloat) { float duration = (this.m_Accordion != null) ? this.m_Accordion.transitionDuration : 0.3f; @@ -124,7 +187,14 @@ namespace UnityEngine.UI.Extensions startFloat = startFloat, targetFloat = targetFloat }; - info.AddOnChangedCallback(SetHeight); + if (m_Accordion.ExpandVerticval) + { + info.AddOnChangedCallback(SetHeight); + } + else + { + info.AddOnChangedCallback(SetWidth); + } info.ignoreTimeScale = true; this.m_FloatTweenRunner.StartTween(info); } @@ -136,5 +206,13 @@ namespace UnityEngine.UI.Extensions this.m_LayoutElement.preferredHeight = height; } + + protected void SetWidth(float width) + { + if (this.m_LayoutElement == null) + return; + + this.m_LayoutElement.preferredWidth = width; + } } } \ No newline at end of file