// Copyright (c) ppy Pty Ltd . Licensed under the MIT Licence. // See the LICENCE file in the repository root for full licence text. using osu.Framework.Allocation; using osu.Framework.Bindables; using osu.Framework.Graphics; using osu.Framework.Graphics.Containers; using osu.Framework.Graphics.UserInterface; using osu.Game.Beatmaps; using osu.Game.Overlays.Settings; namespace osu.Game.Rulesets.Mods { public class DifficultyAdjustSettingsControl : SettingsItem { [Resolved] private IBindable beatmap { get; set; } /// /// Used to track the display value on the setting slider. /// This can either be a user override or the beatmap default (when is null). /// private readonly BindableNumber displayNumber = new BindableNumber(); protected override Drawable CreateControl() => new ControlDrawable(displayNumber); private bool isInternalChange; private DifficultyBindable difficultyBindable; public override Bindable Current { get => base.Current; set { // intercept and extract the DifficultyBindable. difficultyBindable = (DifficultyBindable)value; // this bind is used to transfer bounds/precision only. displayNumber.BindTo(difficultyBindable.CurrentNumber); base.Current = value; } } protected override void LoadComplete() { base.LoadComplete(); beatmap.BindValueChanged(b => updateFromDifficulty(), true); Current.BindValueChanged(current => { // the user override has changed; transfer the correct value to the visual display. if (current.NewValue == null) updateFromDifficulty(); else displayNumber.Value = current.NewValue.Value; }); displayNumber.BindValueChanged(number => { if (!isInternalChange) Current.Value = number.NewValue; }); } private void updateFromDifficulty() { var difficulty = beatmap.Value.BeatmapInfo.BaseDifficulty; if (difficulty == null) return; if (Current.Value == null) { // ensure the beatmap's value is not transferred as a user override. isInternalChange = true; displayNumber.Value = difficultyBindable.ReadFromDifficulty(difficulty); isInternalChange = false; } } private class ControlDrawable : CompositeDrawable, IHasCurrentValue { private readonly BindableWithCurrent current = new BindableWithCurrent(); // Mainly just for fulfilling the interface requirements. // The actual update flow is done via the provided number. public Bindable Current { get => current.Current; set => current.Current = value; } public ControlDrawable(BindableNumber currentNumber) { InternalChildren = new Drawable[] { new SettingsSlider { ShowsDefaultIndicator = false, Current = currentNumber, } }; AutoSizeAxes = Axes.Y; RelativeSizeAxes = Axes.X; } } } }