2019-01-24 16:43:03 +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.
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
|
|
|
|
using System;
|
|
|
|
|
using System.Globalization;
|
2018-11-20 15:51:59 +08:00
|
|
|
|
using osuTK;
|
|
|
|
|
using osuTK.Graphics;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
using osu.Framework.Allocation;
|
|
|
|
|
using osu.Framework.Audio;
|
|
|
|
|
using osu.Framework.Audio.Sample;
|
|
|
|
|
using osu.Framework.Graphics;
|
|
|
|
|
using osu.Framework.Graphics.UserInterface;
|
|
|
|
|
using osu.Framework.Graphics.Cursor;
|
|
|
|
|
using osu.Framework.Graphics.Shapes;
|
2018-10-02 11:02:47 +08:00
|
|
|
|
using osu.Framework.Input.Events;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
|
|
|
|
namespace osu.Game.Graphics.UserInterface
|
|
|
|
|
{
|
|
|
|
|
public class OsuSliderBar<T> : SliderBar<T>, IHasTooltip, IHasAccentColour
|
|
|
|
|
where T : struct, IEquatable<T>, IComparable, IConvertible
|
|
|
|
|
{
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Maximum number of decimal digits to be displayed in the tooltip.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private const int max_decimal_digits = 5;
|
|
|
|
|
|
|
|
|
|
private SampleChannel sample;
|
|
|
|
|
private double lastSampleTime;
|
|
|
|
|
private T lastSampleValue;
|
|
|
|
|
|
|
|
|
|
protected readonly Nub Nub;
|
|
|
|
|
private readonly Box leftBox;
|
|
|
|
|
private readonly Box rightBox;
|
|
|
|
|
|
2019-01-08 15:07:54 +08:00
|
|
|
|
public virtual string TooltipText { get; private set; }
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
|
|
|
|
private Color4 accentColour;
|
2019-02-28 12:31:40 +08:00
|
|
|
|
|
2018-04-13 17:19:50 +08:00
|
|
|
|
public Color4 AccentColour
|
|
|
|
|
{
|
2019-02-28 12:58:19 +08:00
|
|
|
|
get => accentColour;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
set
|
|
|
|
|
{
|
|
|
|
|
accentColour = value;
|
|
|
|
|
leftBox.Colour = value;
|
|
|
|
|
rightBox.Colour = value;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public OsuSliderBar()
|
|
|
|
|
{
|
|
|
|
|
Height = 12;
|
|
|
|
|
RangePadding = 20;
|
|
|
|
|
Children = new Drawable[]
|
|
|
|
|
{
|
|
|
|
|
leftBox = new Box
|
|
|
|
|
{
|
|
|
|
|
Height = 2,
|
|
|
|
|
EdgeSmoothness = new Vector2(0, 0.5f),
|
|
|
|
|
Position = new Vector2(2, 0),
|
|
|
|
|
RelativeSizeAxes = Axes.None,
|
|
|
|
|
Anchor = Anchor.CentreLeft,
|
|
|
|
|
Origin = Anchor.CentreLeft,
|
|
|
|
|
},
|
|
|
|
|
rightBox = new Box
|
|
|
|
|
{
|
|
|
|
|
Height = 2,
|
|
|
|
|
EdgeSmoothness = new Vector2(0, 0.5f),
|
|
|
|
|
Position = new Vector2(-2, 0),
|
|
|
|
|
RelativeSizeAxes = Axes.None,
|
|
|
|
|
Anchor = Anchor.CentreRight,
|
|
|
|
|
Origin = Anchor.CentreRight,
|
|
|
|
|
Alpha = 0.5f,
|
|
|
|
|
},
|
|
|
|
|
Nub = new Nub
|
|
|
|
|
{
|
|
|
|
|
Origin = Anchor.TopCentre,
|
|
|
|
|
Expanded = true,
|
|
|
|
|
},
|
|
|
|
|
new HoverClickSounds()
|
|
|
|
|
};
|
|
|
|
|
|
2019-02-28 12:31:40 +08:00
|
|
|
|
Current.DisabledChanged += disabled => { Alpha = disabled ? 0.3f : 1; };
|
2018-04-13 17:19:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[BackgroundDependencyLoader]
|
2018-08-31 06:04:40 +08:00
|
|
|
|
private void load(AudioManager audio, OsuColour colours)
|
2018-04-13 17:19:50 +08:00
|
|
|
|
{
|
2019-05-28 16:06:01 +08:00
|
|
|
|
sample = audio.Samples.Get(@"UI/sliderbar-notch");
|
2018-04-13 17:19:50 +08:00
|
|
|
|
AccentColour = colours.Pink;
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-17 16:03:53 +08:00
|
|
|
|
protected override void LoadComplete()
|
|
|
|
|
{
|
|
|
|
|
base.LoadComplete();
|
2019-02-22 19:13:38 +08:00
|
|
|
|
CurrentNumber.BindValueChanged(current => updateTooltipText(current.NewValue), true);
|
2019-01-17 16:03:53 +08:00
|
|
|
|
}
|
|
|
|
|
|
2018-10-02 11:02:47 +08:00
|
|
|
|
protected override bool OnHover(HoverEvent e)
|
2018-04-13 17:19:50 +08:00
|
|
|
|
{
|
|
|
|
|
Nub.Glowing = true;
|
2018-10-02 11:02:47 +08:00
|
|
|
|
return base.OnHover(e);
|
2018-04-13 17:19:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
2018-10-02 11:02:47 +08:00
|
|
|
|
protected override void OnHoverLost(HoverLostEvent e)
|
2018-04-13 17:19:50 +08:00
|
|
|
|
{
|
|
|
|
|
Nub.Glowing = false;
|
2018-10-02 11:02:47 +08:00
|
|
|
|
base.OnHoverLost(e);
|
2018-04-13 17:19:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-01-08 15:07:54 +08:00
|
|
|
|
protected override bool OnMouseDown(MouseDownEvent e)
|
2018-04-13 17:19:50 +08:00
|
|
|
|
{
|
2019-01-08 15:07:54 +08:00
|
|
|
|
Nub.Current.Value = true;
|
|
|
|
|
return base.OnMouseDown(e);
|
2018-04-13 17:19:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-01-08 15:07:54 +08:00
|
|
|
|
protected override bool OnMouseUp(MouseUpEvent e)
|
|
|
|
|
{
|
|
|
|
|
Nub.Current.Value = false;
|
|
|
|
|
return base.OnMouseUp(e);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override void OnUserChange(T value)
|
|
|
|
|
{
|
|
|
|
|
base.OnUserChange(value);
|
|
|
|
|
playSample(value);
|
2019-01-23 10:21:09 +08:00
|
|
|
|
updateTooltipText(value);
|
2019-01-08 15:07:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void playSample(T value)
|
2018-04-13 17:19:50 +08:00
|
|
|
|
{
|
|
|
|
|
if (Clock == null || Clock.CurrentTime - lastSampleTime <= 50)
|
|
|
|
|
return;
|
|
|
|
|
|
2019-01-08 15:07:54 +08:00
|
|
|
|
if (value.Equals(lastSampleValue))
|
2018-04-13 17:19:50 +08:00
|
|
|
|
return;
|
|
|
|
|
|
2019-01-08 15:07:54 +08:00
|
|
|
|
lastSampleValue = value;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
|
|
|
|
lastSampleTime = Clock.CurrentTime;
|
|
|
|
|
sample.Frequency.Value = 1 + NormalizedValue * 0.2f;
|
|
|
|
|
|
|
|
|
|
if (NormalizedValue == 0)
|
|
|
|
|
sample.Frequency.Value -= 0.4f;
|
|
|
|
|
else if (NormalizedValue == 1)
|
|
|
|
|
sample.Frequency.Value += 0.4f;
|
|
|
|
|
|
|
|
|
|
sample.Play();
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-08 15:07:54 +08:00
|
|
|
|
private void updateTooltipText(T value)
|
2018-04-13 17:19:50 +08:00
|
|
|
|
{
|
2019-01-08 15:07:54 +08:00
|
|
|
|
if (CurrentNumber.IsInteger)
|
|
|
|
|
TooltipText = ((int)Convert.ChangeType(value, typeof(int))).ToString("N0");
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
double floatValue = (double)Convert.ChangeType(value, typeof(double));
|
|
|
|
|
double floatMinValue = (double)Convert.ChangeType(CurrentNumber.MinValue, typeof(double));
|
|
|
|
|
double floatMaxValue = (double)Convert.ChangeType(CurrentNumber.MaxValue, typeof(double));
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2019-01-08 15:07:54 +08:00
|
|
|
|
if (floatMaxValue == 1 && floatMinValue >= -1)
|
|
|
|
|
TooltipText = floatValue.ToString("P0");
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
var decimalPrecision = normalise((decimal)Convert.ChangeType(CurrentNumber.Precision, typeof(decimal)), max_decimal_digits);
|
|
|
|
|
|
|
|
|
|
// Find the number of significant digits (we could have less than 5 after normalize())
|
|
|
|
|
var significantDigits = findPrecision(decimalPrecision);
|
|
|
|
|
|
|
|
|
|
TooltipText = floatValue.ToString($"N{significantDigits}");
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-04-13 17:19:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override void UpdateAfterChildren()
|
|
|
|
|
{
|
|
|
|
|
base.UpdateAfterChildren();
|
|
|
|
|
leftBox.Scale = new Vector2(MathHelper.Clamp(
|
|
|
|
|
Nub.DrawPosition.X - Nub.DrawWidth / 2, 0, DrawWidth), 1);
|
|
|
|
|
rightBox.Scale = new Vector2(MathHelper.Clamp(
|
|
|
|
|
DrawWidth - Nub.DrawPosition.X - Nub.DrawWidth / 2, 0, DrawWidth), 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override void UpdateValue(float value)
|
|
|
|
|
{
|
|
|
|
|
Nub.MoveToX(RangePadding + UsableWidth * value, 250, Easing.OutQuint);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Removes all non-significant digits, keeping at most a requested number of decimal digits.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="d">The decimal to normalize.</param>
|
|
|
|
|
/// <param name="sd">The maximum number of decimal digits to keep. The final result may have fewer decimal digits than this value.</param>
|
|
|
|
|
/// <returns>The normalised decimal.</returns>
|
|
|
|
|
private decimal normalise(decimal d, int sd)
|
|
|
|
|
=> decimal.Parse(Math.Round(d, sd).ToString(string.Concat("0.", new string('#', sd)), CultureInfo.InvariantCulture), CultureInfo.InvariantCulture);
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Finds the number of digits after the decimal.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="d">The value to find the number of decimal digits for.</param>
|
|
|
|
|
/// <returns>The number decimal digits.</returns>
|
|
|
|
|
private int findPrecision(decimal d)
|
|
|
|
|
{
|
|
|
|
|
int precision = 0;
|
2019-04-01 11:16:05 +08:00
|
|
|
|
|
2018-04-13 17:19:50 +08:00
|
|
|
|
while (d != Math.Round(d))
|
|
|
|
|
{
|
|
|
|
|
d *= 10;
|
|
|
|
|
precision++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return precision;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|