mirror of
https://github.com/Ellpeck/MLEM.git
synced 2024-11-16 02:53:11 +01:00
147 lines
No EOL
7.5 KiB
C#
147 lines
No EOL
7.5 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using MLEM.Ui.Elements;
|
|
|
|
namespace MLEM.Ui.Style {
|
|
/// <summary>
|
|
/// A struct used by <see cref="Element"/> to store style properties.
|
|
/// This is a helper struct that allows default style settings from <see cref="UiStyle"/> to be overridden by custom user settings easily.
|
|
/// Note that <c>T</c> implicitly converts to <c>StyleProp{T}</c> and vice versa.
|
|
/// </summary>
|
|
/// <typeparam name="T">The type of style setting that this property stores</typeparam>
|
|
public readonly struct StyleProp<T> : IEquatable<StyleProp<T>> {
|
|
|
|
/// <summary>
|
|
/// The empty style property, with no <see cref="Value"/> and a priority of 0.
|
|
/// </summary>
|
|
public static StyleProp<T> None => default;
|
|
|
|
/// <summary>
|
|
/// The currently applied style
|
|
/// </summary>
|
|
public readonly T Value;
|
|
private readonly byte priority;
|
|
|
|
/// <summary>
|
|
/// Creates a new style property with the given custom style and a priority of <see cref="byte.MaxValue"/>.
|
|
/// To create a style property with a lower priority, use <see cref="OrStyle(T,byte)"/> on an existing priority, or use <see cref="None"/>.
|
|
/// </summary>
|
|
/// <param name="value">The custom style to apply</param>
|
|
public StyleProp(T value) : this(value, byte.MaxValue) {}
|
|
|
|
private StyleProp(T value, byte priority) {
|
|
this.Value = value;
|
|
this.priority = priority;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates a copy of this style property and sets its value and marks it as being set by a <see cref="UiStyle"/> if it doesn't have a custom value yet.
|
|
/// This allows this property to be overridden by custom style settings using <see cref="StyleProp{T}(T)"/> or a higher <paramref name="priority"/>.
|
|
/// </summary>
|
|
/// <param name="value">The style to apply</param>
|
|
/// <param name="priority">The priority that the style value has. Higher priority style values will override lower priority style values.</param>
|
|
/// <returns>The style with the higher priority</returns>
|
|
public StyleProp<T> OrStyle(T value, byte priority = 0) {
|
|
return this.OrStyle(new StyleProp<T>(value, priority));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Chooses and returns the style property with the higher priority, out of this value and <paramref name="other"/>.
|
|
/// This allows this property to be overridden by custom style settings using <see cref="StyleProp{T}(T)"/> or a higher priority.
|
|
/// </summary>
|
|
/// <param name="other">The style property to compare with</param>
|
|
/// <returns>The style property with the higher priority</returns>
|
|
public StyleProp<T> OrStyle(StyleProp<T> other) {
|
|
return other.priority >= this.priority ? other : this;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns the current style value or, if <see cref="HasValue"/> is false, the given default value.
|
|
/// </summary>
|
|
/// <param name="def">The default to return if this style property has no value</param>
|
|
/// <returns>The current value, or the default</returns>
|
|
public T OrDefault(T def) {
|
|
return this.HasValue() ? this.Value : def;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns whether this style property has a value assigned to it using <see cref="OrStyle(StyleProp{T})"/> or <see cref="StyleProp{T}(T)"/>.
|
|
/// </summary>
|
|
/// <returns>Whether this style property has a value</returns>
|
|
public bool HasValue() {
|
|
return !EqualityComparer<T>.Default.Equals(this.Value, default);
|
|
}
|
|
|
|
/// <summary>Indicates whether the current object is equal to another object of the same type.</summary>
|
|
/// <param name="other">An object to compare with this object.</param>
|
|
/// <returns>true if the current object is equal to the <paramref name="other">other</paramref> parameter; otherwise, false.</returns>
|
|
[Obsolete("StyleProp equality is ambiguous as it is not clear whether priority is taken into account. Compare Values instead.")]
|
|
public bool Equals(StyleProp<T> other) {
|
|
return EqualityComparer<T>.Default.Equals(this.Value, other.Value);
|
|
}
|
|
|
|
/// <summary>Indicates whether this instance and a specified object are equal.</summary>
|
|
/// <param name="obj">The object to compare with the current instance.</param>
|
|
/// <returns>true if <paramref name="obj">obj</paramref> and this instance are the same type and represent the same value; otherwise, false.</returns>
|
|
[Obsolete("StyleProp equality is ambiguous as it is not clear whether priority is taken into account. Compare Values instead.")]
|
|
#pragma warning disable CS0809
|
|
public override bool Equals(object obj) {
|
|
return obj is StyleProp<T> other && this.Equals(other);
|
|
}
|
|
|
|
/// <summary>Returns the hash code for this instance.</summary>
|
|
/// <returns>A 32-bit signed integer that is the hash code for this instance.</returns>
|
|
[Obsolete("StyleProp equality is ambiguous as it is not clear whether priority is taken into account. Compare Values instead.")]
|
|
public override int GetHashCode() {
|
|
return EqualityComparer<T>.Default.GetHashCode(this.Value);
|
|
}
|
|
#pragma warning restore CS0809
|
|
|
|
/// <summary>Returns the fully qualified type name of this instance.</summary>
|
|
/// <returns>The fully qualified type name.</returns>
|
|
public override string ToString() {
|
|
return this.Value?.ToString();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Implicitly converts a style property to its value.
|
|
/// </summary>
|
|
/// <param name="prop">The property to convert</param>
|
|
/// <returns>The style that the style property holds</returns>
|
|
public static implicit operator T(StyleProp<T> prop) {
|
|
return prop.Value;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Implicitly converts a style to a style property.
|
|
/// </summary>
|
|
/// <param name="prop">The property to convert</param>
|
|
/// <returns>A style property with the given style value</returns>
|
|
public static implicit operator StyleProp<T>(T prop) {
|
|
return new StyleProp<T>(prop);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Compares the two style properties and returns whether they are equal using <see cref="Equals(MLEM.Ui.Style.StyleProp{T})"/>.
|
|
/// </summary>
|
|
/// <param name="left">The left style property.</param>
|
|
/// <param name="right">The right style property.</param>
|
|
/// <returns>Whether the two style properties are equal.</returns>
|
|
[Obsolete("StyleProp equality is ambiguous as it is not clear whether priority is taken into account. Compare Values instead.")]
|
|
public static bool operator ==(StyleProp<T> left, StyleProp<T> right) {
|
|
return left.Equals(right);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Compares the two style properties and returns whether they are not equal using <see cref="Equals(MLEM.Ui.Style.StyleProp{T})"/>.
|
|
/// </summary>
|
|
/// <param name="left">The left style property.</param>
|
|
/// <param name="right">The right style property.</param>
|
|
/// <returns>Whether the two style properties are not equal.</returns>
|
|
[Obsolete("StyleProp equality is ambiguous as it is not clear whether priority is taken into account. Compare Values instead.")]
|
|
public static bool operator !=(StyleProp<T> left, StyleProp<T> right) {
|
|
return !left.Equals(right);
|
|
}
|
|
|
|
}
|
|
} |