1
0
Fork 0
mirror of https://github.com/Ellpeck/MLEM.git synced 2024-11-26 06:28:35 +01:00

Added Element.AutoSizeAddedAbsolute to allow for more granular control of auto-sizing

This commit is contained in:
Ell 2022-08-16 14:20:32 +02:00
parent e50d28ce11
commit af0aee6c40
5 changed files with 51 additions and 7 deletions

View file

@ -19,6 +19,7 @@ Improvements
### MLEM.Ui ### MLEM.Ui
Additions Additions
- Added some extension methods for querying Anchor types - Added some extension methods for querying Anchor types
- Added Element.AutoSizeAddedAbsolute to allow for more granular control of auto-sizing
Improvements Improvements
- Allow elements to auto-adjust their size even when their children are aligned oddly - Allow elements to auto-adjust their size even when their children are aligned oddly

2
FNA

@ -1 +1 @@
Subproject commit 102990f514f1e5bfac07d33f7c33e2e712946da4 Subproject commit 31ff5cd8b6d62b0c10b627c67cc66db1d4def4e7

@ -1 +1 @@
Subproject commit f0774130cad6cec0b790a58bc7c811a186443fb3 Subproject commit 7373710db2260a5a0214c8102015b1e11425fa25

View file

@ -71,9 +71,10 @@ namespace MLEM.Ui.Elements {
/// The size of this element, where X represents the width and Y represents the height. /// The size of this element, where X represents the width and Y represents the height.
/// If the x or y value of the size is between 0 and 1, the size will be seen as a percentage of its parent's size rather than as an absolute value. /// If the x or y value of the size is between 0 and 1, the size will be seen as a percentage of its parent's size rather than as an absolute value.
/// If the x (or y) value of the size is negative, the width (or height) is seen as a percentage of the element's resulting height (or width). /// If the x (or y) value of the size is negative, the width (or height) is seen as a percentage of the element's resulting height (or width).
/// Additionally, if auto-sizing is used, <see cref="AutoSizeAddedAbsolute"/> can be set to add or subtract an absolute value from the automatically calculated size.
/// </summary> /// </summary>
/// <example> /// <example>
/// The following example combines both types of percentage-based sizing. /// The following example, ignoring <see cref="Scale"/>, combines both types of percentage-based sizing.
/// If this element is inside of a <see cref="Parent"/> whose width is 20, this element's width will be set to <c>0.5 * 20 = 10</c>, and its height will be set to <c>2.5 * 10 = 25</c>. /// If this element is inside of a <see cref="Parent"/> whose width is 20, this element's width will be set to <c>0.5 * 20 = 10</c>, and its height will be set to <c>2.5 * 10 = 25</c>.
/// <code> /// <code>
/// element.Size = new Vector2(0.5F, -2.5F); /// element.Size = new Vector2(0.5F, -2.5F);
@ -93,6 +94,26 @@ namespace MLEM.Ui.Elements {
/// </summary> /// </summary>
public Vector2 ScaledSize => this.size * this.Scale; public Vector2 ScaledSize => this.size * this.Scale;
/// <summary> /// <summary>
/// If auto-sizing is used by setting <see cref="Size"/> less than or equal to 1, this property allows adding or subtracting an additional, absolute value from the automatically calculated size.
/// If this element is not using auto-sizing, this property is ignored.
/// </summary>
/// <example>
/// Ignoring <see cref="Scale"/>, if this element's <see cref="Size"/> is set to <c>0.5, 0.75</c> and its <see cref="Parent"/> has a size of <c>200, 100</c>, then an added absolute size of <c>-50, 25</c> will result in a final <see cref="Area"/> size of <c>0.5 * 200 - 50, 0.75 * 100 + 25</c>, or <c>50, 100</c>.
/// </example>
public Vector2 AutoSizeAddedAbsolute {
get => this.autoSizeAddedAbsolute;
set {
if (this.autoSizeAddedAbsolute == value)
return;
this.autoSizeAddedAbsolute = value;
this.SetAreaDirty();
}
}
/// <summary>
/// The <see cref="AutoSizeAddedAbsolute"/>, but with <see cref="Scale"/> applied.
/// </summary>
public Vector2 ScaledAutoSizeAddedAbsolute => this.autoSizeAddedAbsolute * this.Scale;
/// <summary>
/// This element's offset from its default position, which is dictated by its <see cref="Anchor"/>. /// This element's offset from its default position, which is dictated by its <see cref="Anchor"/>.
/// Note that, depending on the side that the element is anchored to, this offset moves it in a different direction. /// Note that, depending on the side that the element is anchored to, this offset moves it in a different direction.
/// </summary> /// </summary>
@ -430,6 +451,7 @@ namespace MLEM.Ui.Elements {
private UiSystem system; private UiSystem system;
private Anchor anchor; private Anchor anchor;
private Vector2 size; private Vector2 size;
private Vector2 autoSizeAddedAbsolute;
private Vector2 offset; private Vector2 offset;
private RectangleF area; private RectangleF area;
private bool isHidden; private bool isHidden;
@ -751,12 +773,12 @@ namespace MLEM.Ui.Elements {
/// <returns>The actual size of this element, taking <see cref="Scale"/> into account</returns> /// <returns>The actual size of this element, taking <see cref="Scale"/> into account</returns>
protected virtual Vector2 CalcActualSize(RectangleF parentArea) { protected virtual Vector2 CalcActualSize(RectangleF parentArea) {
var ret = new Vector2( var ret = new Vector2(
this.size.X > 1 ? this.ScaledSize.X : parentArea.Width * this.size.X, this.size.X > 1 ? this.ScaledSize.X : parentArea.Width * this.size.X + this.ScaledAutoSizeAddedAbsolute.X,
this.size.Y > 1 ? this.ScaledSize.Y : parentArea.Height * this.size.Y); this.size.Y > 1 ? this.ScaledSize.Y : parentArea.Height * this.size.Y + this.ScaledAutoSizeAddedAbsolute.Y);
if (this.size.X < 0) if (this.size.X < 0)
ret.X = -this.size.X * ret.Y; ret.X = -this.size.X * ret.Y + this.ScaledAutoSizeAddedAbsolute.X;
if (this.size.Y < 0) if (this.size.Y < 0)
ret.Y = -this.size.Y * ret.X; ret.Y = -this.size.Y * ret.X + this.ScaledAutoSizeAddedAbsolute.Y;
return ret; return ret;
} }

View file

@ -77,6 +77,27 @@ namespace Tests {
Assert.AreEqual(testBtn.GetChildren<Group>().Single().DisplayArea.Width, (panelContentWidth - 3 - 3) * 0.5F); Assert.AreEqual(testBtn.GetChildren<Group>().Single().DisplayArea.Width, (panelContentWidth - 3 - 3) * 0.5F);
} }
[Test]
public void TestAbsoluteAutoSize() {
var parent = new Panel(Anchor.AutoLeft, new Vector2(200, 100), Vector2.Zero) {
ChildPadding = Padding.Empty
};
var el1 = parent.AddChild(new Button(Anchor.AutoLeft, new Vector2(0.5F, 0.75F)) {
AutoSizeAddedAbsolute = new Vector2(-50, 25)
});
var el2 = parent.AddChild(new Button(Anchor.AutoLeft, new Vector2(0.25F, -0.5F)) {
AutoSizeAddedAbsolute = new Vector2(-25, 50)
});
this.AddAndUpdate(parent);
Assert.AreEqual(0.5F * 200 - 50, el1.DisplayArea.Width);
Assert.AreEqual(0.75F * 100 + 25, el1.DisplayArea.Height);
const float el2Width = 0.25F * 200 - 25;
Assert.AreEqual(el2Width, el2.DisplayArea.Width);
Assert.AreEqual(0.5F * el2Width + 50, el2.DisplayArea.Height);
}
[Test] [Test]
public void TestStyle() { public void TestStyle() {
var style = new StyleProp<string>(); var style = new StyleProp<string>();