Updated Accordion to support both Vertical as well as Horizontal layout

release
Simon (darkside) Jackson 2020-08-30 16:09:54 +01:00
parent b2798a6c5e
commit 84bf882777
4 changed files with 181 additions and 38 deletions

View File

@ -711,25 +711,70 @@ namespace UnityEditor.UI
#endregion #endregion
#region Accordion #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)] [MenuItem("GameObject/UI/Extensions/Accordion/Accordion Group", false)]
static public void AddAccordionGroup(MenuCommand menuCommand) static public void AddAccordionGroup(MenuCommand menuCommand)
{ {
GameObject go = CreateUIElementRoot("Accordion Group", menuCommand, s_ThickGUIElementSize); GameObject go = CreateUIElementRoot("Accordion Group", menuCommand, s_ThickGUIElementSize);
go.AddComponent<VerticalLayoutGroup>(); CreateAccordionGroup(go);
go.AddComponent<ContentSizeFitter>(); Selection.activeGameObject = go;
}
private static void CreateAccordionGroup(GameObject go)
{
var vlg = go.AddComponent<VerticalLayoutGroup>();
vlg.childControlHeight = true;
vlg.childControlWidth = true;
vlg.childForceExpandHeight = false;
vlg.childForceExpandHeight = true;
var csf = go.AddComponent<ContentSizeFitter>();
csf.verticalFit = ContentSizeFitter.FitMode.PreferredSize;
go.AddComponent<ToggleGroup>(); go.AddComponent<ToggleGroup>();
go.AddComponent<Accordion>(); go.AddComponent<Accordion>();
Selection.activeGameObject = go;
} }
[MenuItem("GameObject/UI/Extensions/Accordion/Accordion Element", false)] [MenuItem("GameObject/UI/Extensions/Accordion/Accordion Element", false)]
static public void AddAccordionElement(MenuCommand menuCommand) static public void AddAccordionElement(MenuCommand menuCommand)
{ {
GameObject go = CreateUIElementRoot("Accordion Element", menuCommand, s_ThickGUIElementSize); GameObject go = CreateUIElementRoot("Accordion Element", menuCommand, s_ThickGUIElementSize);
go.AddComponent<LayoutElement>(); CreateAccordionElement(go);
go.AddComponent<AccordionElement>();
Selection.activeGameObject = go;
Selection.activeGameObject = go;
}
private static void CreateAccordionElement(GameObject go)
{
var vlg = go.AddComponent<VerticalLayoutGroup>();
vlg.childControlHeight = true;
vlg.childControlWidth = true;
vlg.childForceExpandHeight = false;
vlg.childForceExpandHeight = true;
go.AddComponent<LayoutElement>();
var accordionElement = go.AddComponent<AccordionElement>();
// Header
GameObject headergo = CreateUIObject("Header", go);
var headerLayout = headergo.AddComponent<LayoutElement>();
headerLayout.minHeight = accordionElement.MinHeight;
var headerText = headergo.AddComponent<Text>();
headerText.text = "This is an Accordion header";
// Text
GameObject textgo = CreateUIObject("Text", go);
var textText = textgo.AddComponent<Text>();
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 #endregion

@ -1 +1 @@
Subproject commit 8f7a9de774bcf64109a5e1362650e9f37572588f Subproject commit cf1a370339e25bc8fcdb10b80d0d9616f12ead0c

View File

@ -4,10 +4,13 @@
namespace UnityEngine.UI.Extensions namespace UnityEngine.UI.Extensions
{ {
[RequireComponent(typeof(VerticalLayoutGroup), typeof(ContentSizeFitter), typeof(ToggleGroup))] [RequireComponent(typeof(HorizontalOrVerticalLayoutGroup), typeof(ContentSizeFitter), typeof(ToggleGroup))]
[AddComponentMenu("UI/Extensions/Accordion/Accordion Group")] [AddComponentMenu("UI/Extensions/Accordion/Accordion Group")]
public class Accordion : MonoBehaviour public class Accordion : MonoBehaviour
{ {
private bool m_expandVertical = true;
[HideInInspector]
public bool ExpandVerticval => m_expandVertical;
public enum Transition public enum Transition
{ {
@ -37,5 +40,22 @@ namespace UnityEngine.UI.Extensions
get { return this.m_TransitionDuration; } get { return this.m_TransitionDuration; }
set { this.m_TransitionDuration = value; } set { this.m_TransitionDuration = value; }
} }
private void Awake()
{
m_expandVertical = GetComponent<HorizontalLayoutGroup>() ? false : true;
var group = GetComponent<ToggleGroup>();
}
#if UNITY_EDITOR
private void OnValidate()
{
if (!GetComponent<HorizontalLayoutGroup>() && !GetComponent<VerticalLayoutGroup>())
{
Debug.LogError("Accordion requires either a Horizontal or Vertical Layout group to place children");
}
}
#endif
} }
} }

View File

@ -13,6 +13,12 @@ namespace UnityEngine.UI.Extensions
[SerializeField] private float m_MinHeight = 18f; [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 Accordion m_Accordion;
private RectTransform m_RectTransform; private RectTransform m_RectTransform;
private LayoutElement m_LayoutElement; private LayoutElement m_LayoutElement;
@ -43,6 +49,7 @@ namespace UnityEngine.UI.Extensions
protected override void OnValidate() protected override void OnValidate()
{ {
base.OnValidate(); base.OnValidate();
this.m_Accordion = this.gameObject.GetComponentInParent<Accordion>();
if (this.group == null) if (this.group == null)
{ {
@ -59,13 +66,28 @@ namespace UnityEngine.UI.Extensions
if (le != null) if (le != null)
{ {
if (this.isOn) if (this.isOn)
{
if (m_Accordion.ExpandVerticval)
{ {
le.preferredHeight = -1f; le.preferredHeight = -1f;
} }
else else
{
le.preferredWidth = -1f;
}
}
else
{
if (m_Accordion.ExpandVerticval)
{ {
le.preferredHeight = this.m_MinHeight; le.preferredHeight = this.m_MinHeight;
} }
else
{
le.preferredWidth = this.m_MinWidth;
}
}
} }
} }
#endif #endif
@ -80,24 +102,52 @@ namespace UnityEngine.UI.Extensions
if (transition == Accordion.Transition.Instant) if (transition == Accordion.Transition.Instant)
{ {
if (state) if (state)
{
if (m_Accordion.ExpandVerticval)
{ {
this.m_LayoutElement.preferredHeight = -1f; this.m_LayoutElement.preferredHeight = -1f;
} }
else else
{
this.m_LayoutElement.preferredWidth = -1f;
}
}
else
{
if (m_Accordion.ExpandVerticval)
{ {
this.m_LayoutElement.preferredHeight = this.m_MinHeight; this.m_LayoutElement.preferredHeight = this.m_MinHeight;
} }
else
{
this.m_LayoutElement.preferredWidth = this.m_MinWidth;
}
}
} }
else if (transition == Accordion.Transition.Tween) else if (transition == Accordion.Transition.Tween)
{ {
if (state) if (state)
{
if (m_Accordion.ExpandVerticval)
{ {
this.StartTween(this.m_MinHeight, this.GetExpandedHeight()); this.StartTween(this.m_MinHeight, this.GetExpandedHeight());
} }
else else
{
this.StartTween(this.m_MinWidth, this.GetExpandedWidth());
}
}
else
{
if (m_Accordion.ExpandVerticval)
{ {
this.StartTween(this.m_RectTransform.rect.height, this.m_MinHeight); this.StartTween(this.m_RectTransform.rect.height, this.m_MinHeight);
} }
else
{
this.StartTween(this.m_RectTransform.rect.width, this.m_MinWidth);
}
}
} }
} }
@ -114,6 +164,19 @@ namespace UnityEngine.UI.Extensions
return h; 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) protected void StartTween(float startFloat, float targetFloat)
{ {
float duration = (this.m_Accordion != null) ? this.m_Accordion.transitionDuration : 0.3f; float duration = (this.m_Accordion != null) ? this.m_Accordion.transitionDuration : 0.3f;
@ -124,7 +187,14 @@ namespace UnityEngine.UI.Extensions
startFloat = startFloat, startFloat = startFloat,
targetFloat = targetFloat targetFloat = targetFloat
}; };
if (m_Accordion.ExpandVerticval)
{
info.AddOnChangedCallback(SetHeight); info.AddOnChangedCallback(SetHeight);
}
else
{
info.AddOnChangedCallback(SetWidth);
}
info.ignoreTimeScale = true; info.ignoreTimeScale = true;
this.m_FloatTweenRunner.StartTween(info); this.m_FloatTweenRunner.StartTween(info);
} }
@ -136,5 +206,13 @@ namespace UnityEngine.UI.Extensions
this.m_LayoutElement.preferredHeight = height; this.m_LayoutElement.preferredHeight = height;
} }
protected void SetWidth(float width)
{
if (this.m_LayoutElement == null)
return;
this.m_LayoutElement.preferredWidth = width;
}
} }
} }