2019-01-24 08:43:03 +00: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.
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2018-11-20 07:51:59 +00:00
|
|
|
|
using osuTK;
|
|
|
|
|
using osuTK.Graphics;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
using osu.Framework.Allocation;
|
|
|
|
|
using osu.Framework.Extensions.Color4Extensions;
|
|
|
|
|
using osu.Framework.Graphics;
|
|
|
|
|
using osu.Framework.Graphics.Containers;
|
|
|
|
|
using osu.Game.Graphics;
|
|
|
|
|
using osu.Game.Graphics.Sprites;
|
|
|
|
|
using osu.Game.Graphics.UserInterface;
|
|
|
|
|
using osu.Game.Beatmaps;
|
2019-12-12 15:41:46 +00:00
|
|
|
|
using osu.Framework.Bindables;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using osu.Game.Rulesets.Mods;
|
|
|
|
|
using System.Linq;
|
2019-12-20 09:00:04 +00:00
|
|
|
|
using osu.Framework.Threading;
|
|
|
|
|
using osu.Game.Configuration;
|
|
|
|
|
using osu.Game.Overlays.Settings;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
|
|
|
|
namespace osu.Game.Screens.Select.Details
|
|
|
|
|
{
|
|
|
|
|
public class AdvancedStats : Container
|
|
|
|
|
{
|
2019-12-12 15:41:46 +00:00
|
|
|
|
[Resolved]
|
|
|
|
|
private IBindable<IReadOnlyList<Mod>> mods { get; set; }
|
|
|
|
|
|
2018-04-13 09:19:50 +00:00
|
|
|
|
private readonly StatisticRow firstValue, hpDrain, accuracy, approachRate, starDifficulty;
|
|
|
|
|
|
|
|
|
|
private BeatmapInfo beatmap;
|
2019-02-28 04:31:40 +00:00
|
|
|
|
|
2018-04-13 09:19:50 +00:00
|
|
|
|
public BeatmapInfo Beatmap
|
|
|
|
|
{
|
2019-02-28 04:58:19 +00:00
|
|
|
|
get => beatmap;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
set
|
|
|
|
|
{
|
|
|
|
|
if (value == beatmap) return;
|
2019-02-28 04:31:40 +00:00
|
|
|
|
|
2018-04-13 09:19:50 +00:00
|
|
|
|
beatmap = value;
|
|
|
|
|
|
2019-12-12 15:41:46 +00:00
|
|
|
|
updateStatistics();
|
2018-04-13 09:19:50 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public AdvancedStats()
|
|
|
|
|
{
|
|
|
|
|
Child = new FillFlowContainer
|
|
|
|
|
{
|
|
|
|
|
RelativeSizeAxes = Axes.X,
|
|
|
|
|
AutoSizeAxes = Axes.Y,
|
|
|
|
|
Spacing = new Vector2(4f),
|
|
|
|
|
Children = new[]
|
|
|
|
|
{
|
|
|
|
|
firstValue = new StatisticRow(), //circle size/key amount
|
|
|
|
|
hpDrain = new StatisticRow { Title = "HP Drain" },
|
|
|
|
|
accuracy = new StatisticRow { Title = "Accuracy" },
|
|
|
|
|
approachRate = new StatisticRow { Title = "Approach Rate" },
|
|
|
|
|
starDifficulty = new StatisticRow(10, true) { Title = "Star Difficulty" },
|
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[BackgroundDependencyLoader]
|
2019-12-13 01:39:54 +00:00
|
|
|
|
private void load(OsuColour colours)
|
2018-04-13 09:19:50 +00:00
|
|
|
|
{
|
|
|
|
|
starDifficulty.AccentColour = colours.Yellow;
|
2019-12-18 08:41:30 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override void LoadComplete()
|
|
|
|
|
{
|
|
|
|
|
base.LoadComplete();
|
|
|
|
|
|
2019-12-20 09:00:04 +00:00
|
|
|
|
mods.BindValueChanged(modsChanged, true);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private readonly List<ISettingsItem> references = new List<ISettingsItem>();
|
|
|
|
|
|
|
|
|
|
private void modsChanged(ValueChangedEvent<IReadOnlyList<Mod>> mods)
|
|
|
|
|
{
|
|
|
|
|
// TODO: find a more permanent solution for this if/when it is needed in other components.
|
|
|
|
|
// this is generating drawables for the only purpose of storing bindable references.
|
|
|
|
|
foreach (var r in references)
|
|
|
|
|
r.Dispose();
|
|
|
|
|
|
|
|
|
|
references.Clear();
|
|
|
|
|
|
|
|
|
|
ScheduledDelegate debounce = null;
|
|
|
|
|
|
|
|
|
|
foreach (var mod in mods.NewValue.OfType<IApplicableToDifficulty>())
|
|
|
|
|
{
|
|
|
|
|
foreach (var setting in mod.CreateSettingsControls().OfType<ISettingsItem>())
|
|
|
|
|
{
|
|
|
|
|
setting.SettingChanged += () =>
|
|
|
|
|
{
|
|
|
|
|
debounce?.Cancel();
|
|
|
|
|
debounce = Scheduler.AddDelayed(updateStatistics, 100);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
references.Add(setting);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
updateStatistics();
|
2019-12-12 15:41:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void updateStatistics()
|
|
|
|
|
{
|
2019-12-18 08:12:41 +00:00
|
|
|
|
BeatmapDifficulty baseDifficulty = Beatmap?.BaseDifficulty;
|
|
|
|
|
BeatmapDifficulty adjustedDifficulty = null;
|
2019-12-12 15:41:46 +00:00
|
|
|
|
|
2019-12-18 08:00:35 +00:00
|
|
|
|
if (baseDifficulty != null && mods.Value.Any(m => m is IApplicableToDifficulty))
|
2019-12-12 15:41:46 +00:00
|
|
|
|
{
|
2019-12-18 08:12:41 +00:00
|
|
|
|
adjustedDifficulty = baseDifficulty.Clone();
|
2019-12-12 15:41:46 +00:00
|
|
|
|
|
|
|
|
|
foreach (var mod in mods.Value.OfType<IApplicableToDifficulty>())
|
2019-12-18 08:00:35 +00:00
|
|
|
|
mod.ApplyToDifficulty(adjustedDifficulty);
|
2019-12-12 15:41:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-01-28 08:21:24 +00:00
|
|
|
|
switch (Beatmap?.Ruleset?.ID ?? 0)
|
|
|
|
|
{
|
|
|
|
|
case 3:
|
|
|
|
|
// Account for mania differences locally for now
|
|
|
|
|
// Eventually this should be handled in a more modular way, allowing rulesets to return arbitrary difficulty attributes
|
|
|
|
|
firstValue.Title = "Key Count";
|
|
|
|
|
firstValue.Value = (baseDifficulty?.CircleSize ?? 0, null);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
firstValue.Title = "Circle Size";
|
|
|
|
|
firstValue.Value = (baseDifficulty?.CircleSize ?? 0, adjustedDifficulty?.CircleSize);
|
|
|
|
|
break;
|
|
|
|
|
}
|
2019-12-12 15:41:46 +00:00
|
|
|
|
|
2019-12-18 08:12:41 +00:00
|
|
|
|
starDifficulty.Value = ((float)(Beatmap?.StarDifficulty ?? 0), null);
|
2019-12-13 01:39:54 +00:00
|
|
|
|
|
2019-12-18 08:12:41 +00:00
|
|
|
|
hpDrain.Value = (baseDifficulty?.DrainRate ?? 0, adjustedDifficulty?.DrainRate);
|
|
|
|
|
accuracy.Value = (baseDifficulty?.OverallDifficulty ?? 0, adjustedDifficulty?.OverallDifficulty);
|
|
|
|
|
approachRate.Value = (baseDifficulty?.ApproachRate ?? 0, adjustedDifficulty?.ApproachRate);
|
2018-04-13 09:19:50 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private class StatisticRow : Container, IHasAccentColour
|
|
|
|
|
{
|
|
|
|
|
private const float value_width = 25;
|
|
|
|
|
private const float name_width = 70;
|
|
|
|
|
|
|
|
|
|
private readonly float maxValue;
|
|
|
|
|
private readonly bool forceDecimalPlaces;
|
2019-12-18 08:12:41 +00:00
|
|
|
|
private readonly OsuSpriteText name, valueText;
|
2019-12-13 01:39:54 +00:00
|
|
|
|
private readonly Bar bar, modBar;
|
|
|
|
|
|
|
|
|
|
[Resolved]
|
|
|
|
|
private OsuColour colours { get; set; }
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
|
|
|
|
public string Title
|
|
|
|
|
{
|
2019-02-28 04:58:19 +00:00
|
|
|
|
get => name.Text;
|
|
|
|
|
set => name.Text = value;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-12-18 08:12:41 +00:00
|
|
|
|
private (float baseValue, float? adjustedValue) value;
|
2019-12-13 01:39:54 +00:00
|
|
|
|
|
2019-12-18 08:12:41 +00:00
|
|
|
|
public (float baseValue, float? adjustedValue) Value
|
2018-04-13 09:19:50 +00:00
|
|
|
|
{
|
2019-12-18 08:12:41 +00:00
|
|
|
|
get => value;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
set
|
|
|
|
|
{
|
2019-12-18 08:12:41 +00:00
|
|
|
|
if (value == this.value)
|
|
|
|
|
return;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2019-12-18 08:12:41 +00:00
|
|
|
|
this.value = value;
|
|
|
|
|
|
|
|
|
|
bar.Length = value.baseValue / maxValue;
|
|
|
|
|
|
|
|
|
|
valueText.Text = (value.adjustedValue ?? value.baseValue).ToString(forceDecimalPlaces ? "0.00" : "0.##");
|
|
|
|
|
modBar.Length = (value.adjustedValue ?? 0) / maxValue;
|
|
|
|
|
|
|
|
|
|
if (value.adjustedValue > value.baseValue)
|
|
|
|
|
modBar.AccentColour = valueText.Colour = colours.Red;
|
|
|
|
|
else if (value.adjustedValue < value.baseValue)
|
|
|
|
|
modBar.AccentColour = valueText.Colour = colours.BlueDark;
|
2019-12-13 01:39:54 +00:00
|
|
|
|
else
|
2019-12-18 08:12:41 +00:00
|
|
|
|
modBar.AccentColour = valueText.Colour = Color4.White;
|
2019-12-13 01:39:54 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-04-13 09:19:50 +00:00
|
|
|
|
public Color4 AccentColour
|
|
|
|
|
{
|
2019-02-28 04:58:19 +00:00
|
|
|
|
get => bar.AccentColour;
|
|
|
|
|
set => bar.AccentColour = value;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public StatisticRow(float maxValue = 10, bool forceDecimalPlaces = false)
|
|
|
|
|
{
|
|
|
|
|
this.maxValue = maxValue;
|
|
|
|
|
this.forceDecimalPlaces = forceDecimalPlaces;
|
|
|
|
|
RelativeSizeAxes = Axes.X;
|
|
|
|
|
AutoSizeAxes = Axes.Y;
|
|
|
|
|
|
|
|
|
|
Children = new Drawable[]
|
|
|
|
|
{
|
|
|
|
|
new Container
|
|
|
|
|
{
|
|
|
|
|
Width = name_width,
|
|
|
|
|
AutoSizeAxes = Axes.Y,
|
|
|
|
|
Child = name = new OsuSpriteText
|
|
|
|
|
{
|
2019-02-12 04:04:46 +00:00
|
|
|
|
Font = OsuFont.GetFont(size: 13)
|
2018-04-13 09:19:50 +00:00
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
bar = new Bar
|
|
|
|
|
{
|
|
|
|
|
Origin = Anchor.CentreLeft,
|
|
|
|
|
Anchor = Anchor.CentreLeft,
|
|
|
|
|
RelativeSizeAxes = Axes.X,
|
|
|
|
|
Height = 5,
|
|
|
|
|
BackgroundColour = Color4.White.Opacity(0.5f),
|
|
|
|
|
Padding = new MarginPadding { Left = name_width + 10, Right = value_width + 10 },
|
|
|
|
|
},
|
2019-12-13 01:39:54 +00:00
|
|
|
|
modBar = new Bar
|
|
|
|
|
{
|
|
|
|
|
Origin = Anchor.CentreLeft,
|
|
|
|
|
Anchor = Anchor.CentreLeft,
|
|
|
|
|
RelativeSizeAxes = Axes.X,
|
|
|
|
|
Alpha = 0.5f,
|
|
|
|
|
Height = 5,
|
|
|
|
|
Padding = new MarginPadding { Left = name_width + 10, Right = value_width + 10 },
|
|
|
|
|
},
|
2018-04-13 09:19:50 +00:00
|
|
|
|
new Container
|
|
|
|
|
{
|
|
|
|
|
Anchor = Anchor.TopRight,
|
|
|
|
|
Origin = Anchor.TopRight,
|
|
|
|
|
Width = value_width,
|
|
|
|
|
RelativeSizeAxes = Axes.Y,
|
2019-12-18 08:12:41 +00:00
|
|
|
|
Child = valueText = new OsuSpriteText
|
2018-04-13 09:19:50 +00:00
|
|
|
|
{
|
|
|
|
|
Anchor = Anchor.Centre,
|
|
|
|
|
Origin = Anchor.Centre,
|
2019-02-12 04:04:46 +00:00
|
|
|
|
Font = OsuFont.GetFont(size: 13)
|
2018-04-13 09:19:50 +00:00
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|