1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-23 15:13:01 +08:00
osu-lazer/osu.Game/Rulesets/Mods/DifficultyBindable.cs

188 lines
6.4 KiB
C#
Raw Normal View History

2021-07-08 15:40:32 +08:00
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
// See the LICENCE file in the repository root for full licence text.
using System;
using osu.Framework.Bindables;
using osu.Game.Beatmaps;
2021-07-08 15:40:32 +08:00
namespace osu.Game.Rulesets.Mods
{
public class DifficultyBindable : Bindable<float?>
{
/// <summary>
/// Whether the extended limits should be applied to this bindable.
/// </summary>
2021-07-09 12:24:26 +08:00
public readonly BindableBool ExtendedLimits = new BindableBool();
2021-07-08 15:40:32 +08:00
/// <summary>
/// An internal numeric bindable to hold and propagate min/max/precision.
/// The value of this bindable should not be set.
/// </summary>
2021-07-12 10:26:30 +08:00
internal readonly BindableFloat CurrentNumber = new BindableFloat
2021-07-08 15:40:32 +08:00
{
MinValue = 0,
MaxValue = 10,
};
/// <summary>
/// A function that can extract the current value of this setting from a beatmap difficulty for display purposes.
/// </summary>
public Func<IBeatmapDifficultyInfo, float>? ReadCurrentFromDifficulty;
2021-07-08 15:40:32 +08:00
public float Precision
{
set => CurrentNumber.Precision = value;
}
private float minValue;
2021-07-08 15:40:32 +08:00
public float MinValue
{
get => minValue;
set
{
if (value == minValue)
return;
minValue = value;
2023-09-07 13:19:15 +08:00
updateExtents();
}
2021-07-08 15:40:32 +08:00
}
private float maxValue = 10; // matches default max value of `CurrentNumber`
2021-07-08 15:40:32 +08:00
public float MaxValue
{
get => maxValue;
2021-07-08 15:40:32 +08:00
set
{
if (value == maxValue)
return;
maxValue = value;
2023-09-07 13:19:15 +08:00
updateExtents();
2021-07-08 15:40:32 +08:00
}
}
private float? extendedMinValue;
/// <summary>
/// The minimum value to be used when extended limits are applied.
/// </summary>
public float? ExtendedMinValue
{
get => extendedMinValue;
set
{
if (value == extendedMinValue)
return;
extendedMinValue = value;
2023-09-07 13:19:15 +08:00
updateExtents();
}
}
2021-07-08 15:40:32 +08:00
private float? extendedMaxValue;
/// <summary>
/// The maximum value to be used when extended limits are applied.
/// </summary>
public float? ExtendedMaxValue
{
get => extendedMaxValue;
2021-07-08 15:40:32 +08:00
set
{
if (value == extendedMaxValue)
return;
extendedMaxValue = value;
2023-09-07 13:19:15 +08:00
updateExtents();
2021-07-08 15:40:32 +08:00
}
}
public DifficultyBindable()
: this(null)
{
}
public DifficultyBindable(float? defaultValue = null)
: base(defaultValue)
2021-07-08 15:40:32 +08:00
{
2023-09-07 13:19:15 +08:00
ExtendedLimits.BindValueChanged(_ => updateExtents());
}
2021-07-08 15:40:32 +08:00
public override float? Value
{
get => base.Value;
set
2021-07-08 15:40:32 +08:00
{
// Ensure that in the case serialisation runs in the wrong order (and limit extensions aren't applied yet) the deserialised value is still propagated.
if (value != null)
{
CurrentNumber.MinValue = Math.Clamp(MathF.Min(CurrentNumber.MinValue, value.Value), ExtendedMinValue ?? MinValue, MinValue);
CurrentNumber.MaxValue = Math.Clamp(MathF.Max(CurrentNumber.MaxValue, value.Value), MaxValue, ExtendedMaxValue ?? MaxValue);
base.Value = Math.Clamp(value.Value, CurrentNumber.MinValue, CurrentNumber.MaxValue);
}
else
base.Value = value;
}
2021-07-08 15:40:32 +08:00
}
2023-09-07 13:19:15 +08:00
private void updateExtents()
{
CurrentNumber.MinValue = ExtendedLimits.Value && extendedMinValue != null ? extendedMinValue.Value : minValue;
2021-07-08 15:40:32 +08:00
CurrentNumber.MaxValue = ExtendedLimits.Value && extendedMaxValue != null ? extendedMaxValue.Value : maxValue;
}
public override void CopyTo(Bindable<float?> them)
{
if (!(them is DifficultyBindable otherDifficultyBindable))
throw new InvalidOperationException($"Cannot copy to a non-{nameof(DifficultyBindable)}.");
base.CopyTo(them);
otherDifficultyBindable.ReadCurrentFromDifficulty = ReadCurrentFromDifficulty;
// the following max value copies are only safe as long as these values are effectively constants.
otherDifficultyBindable.MaxValue = maxValue;
otherDifficultyBindable.ExtendedMaxValue = extendedMaxValue;
otherDifficultyBindable.MinValue = minValue;
otherDifficultyBindable.ExtendedMinValue = extendedMinValue;
}
public override void BindTo(Bindable<float?> them)
{
if (!(them is DifficultyBindable otherDifficultyBindable))
throw new InvalidOperationException($"Cannot bind to a non-{nameof(DifficultyBindable)}.");
// ensure that MaxValue and ExtendedMaxValue are copied across first before continuing.
// not doing so may cause the value of CurrentNumber to be truncated to 10.
otherDifficultyBindable.CopyTo(this);
// set up mutual binding for ExtendedLimits to correctly set the upper bound of CurrentNumber.
ExtendedLimits.BindTarget = otherDifficultyBindable.ExtendedLimits;
// set up mutual binding for CurrentNumber. this must happen after all of the above.
CurrentNumber.BindTarget = otherDifficultyBindable.CurrentNumber;
2022-11-24 12:57:43 +08:00
// finish up the binding by setting up weak references via the base call.
// unfortunately this will call `.CopyTo()` again, but fixing that is problematic and messy.
2022-11-24 12:57:43 +08:00
base.BindTo(them);
}
2021-07-12 16:33:29 +08:00
public override void UnbindFrom(IUnbindable them)
{
if (!(them is DifficultyBindable otherDifficultyBindable))
throw new InvalidOperationException($"Cannot unbind from a non-{nameof(DifficultyBindable)}.");
base.UnbindFrom(them);
CurrentNumber.UnbindFrom(otherDifficultyBindable.CurrentNumber);
ExtendedLimits.UnbindFrom(otherDifficultyBindable.ExtendedLimits);
}
protected override Bindable<float?> CreateInstance() => new DifficultyBindable();
2021-07-08 15:40:32 +08:00
}
}