osu/osu.Game/Graphics/UserInterface/RollingCounter.cs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

182 lines
6.1 KiB
C#
Raw Normal View History

// 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
2022-06-17 07:37:17 +00:00
#nullable disable
using osu.Framework.Graphics;
2016-10-07 07:05:02 +00:00
using osu.Framework.Graphics.Containers;
2016-10-13 22:13:20 +00:00
using osu.Framework.Graphics.Sprites;
2017-03-10 04:08:59 +00:00
using osu.Game.Graphics.Sprites;
2016-10-07 07:05:02 +00:00
using System;
using System.Collections.Generic;
using osu.Framework.Allocation;
2019-02-21 10:04:31 +00:00
using osu.Framework.Bindables;
using osu.Framework.Graphics.UserInterface;
2021-07-23 20:37:08 +00:00
using osu.Framework.Localisation;
2018-04-13 09:19:50 +00:00
2016-10-07 07:05:02 +00:00
namespace osu.Game.Graphics.UserInterface
{
public abstract partial class RollingCounter<T> : Container, IHasCurrentValue<T>
2017-07-14 13:46:00 +00:00
where T : struct, IEquatable<T>
2016-10-07 07:05:02 +00:00
{
private readonly BindableWithCurrent<T> current = new BindableWithCurrent<T>();
public Bindable<T> Current
{
get => current.Current;
set => current.Current = value;
}
2018-04-13 09:19:50 +00:00
private IHasText displayedCountText;
public Drawable DrawableCount { get; private set; }
2018-04-13 09:19:50 +00:00
2016-10-07 07:05:02 +00:00
/// <summary>
2016-10-14 00:50:06 +00:00
/// If true, the roll-up duration will be proportional to change in value.
2016-10-07 07:05:02 +00:00
/// </summary>
2016-10-18 02:40:50 +00:00
protected virtual bool IsRollingProportional => false;
2018-04-13 09:19:50 +00:00
2016-10-07 07:05:02 +00:00
/// <summary>
2016-10-09 19:48:24 +00:00
/// If IsRollingProportional = false, duration in milliseconds for the counter roll-up animation for each
/// element; else duration in milliseconds for the counter roll-up animation in total.
2016-10-07 07:05:02 +00:00
/// </summary>
2016-10-18 02:40:50 +00:00
protected virtual double RollingDuration => 0;
2018-04-13 09:19:50 +00:00
2016-10-07 07:05:02 +00:00
/// <summary>
/// Easing for the counter rollover animation.
/// </summary>
protected virtual Easing RollingEasing => Easing.OutQuad;
2018-04-13 09:19:50 +00:00
2016-10-16 00:07:07 +00:00
private T displayedCount;
2018-04-13 09:19:50 +00:00
2016-10-07 07:05:02 +00:00
/// <summary>
/// Value shown at the current moment.
/// </summary>
2016-10-16 00:07:07 +00:00
public virtual T DisplayedCount
2016-10-07 07:05:02 +00:00
{
get => displayedCount;
2017-07-14 16:14:07 +00:00
set
2016-10-07 07:05:02 +00:00
{
if (EqualityComparer<T>.Default.Equals(displayedCount, value))
2016-10-07 07:05:02 +00:00
return;
2019-02-28 04:31:40 +00:00
2016-10-16 00:07:07 +00:00
displayedCount = value;
UpdateDisplay();
2016-10-07 07:05:02 +00:00
}
}
2018-04-13 09:19:50 +00:00
2016-10-13 22:13:20 +00:00
/// <summary>
/// Skeleton of a numeric counter which value rolls over time.
/// </summary>
2016-10-13 02:46:51 +00:00
protected RollingCounter()
{
2016-10-22 08:50:42 +00:00
AutoSizeAxes = Axes.Both;
2016-10-07 07:05:02 +00:00
}
2018-04-13 09:19:50 +00:00
[BackgroundDependencyLoader]
private void load()
{
displayedCountText = CreateText();
UpdateDisplay();
Child = DrawableCount = (Drawable)displayedCountText;
}
2018-04-13 09:19:50 +00:00
protected void UpdateDisplay()
{
if (displayedCountText != null)
displayedCountText.Text = FormatCount(DisplayedCount);
}
protected override void LoadComplete()
{
base.LoadComplete();
Current.BindValueChanged(val => TransformCount(DisplayedCount, val.NewValue), true);
}
/// <summary>
/// Sets count value, bypassing rollover animation.
/// </summary>
/// <param name="count">New count value.</param>
public virtual void SetCountWithoutRolling(T count)
{
2017-03-10 04:08:59 +00:00
Current.Value = count;
StopRolling();
}
2018-04-13 09:19:50 +00:00
/// <summary>
2016-10-16 00:07:07 +00:00
/// Stops rollover animation, forcing the displayed count to be the actual count.
/// </summary>
public virtual void StopRolling()
{
2017-07-21 15:24:09 +00:00
FinishTransforms(false, nameof(DisplayedCount));
2019-02-21 09:56:34 +00:00
DisplayedCount = Current.Value;
}
2018-04-13 09:19:50 +00:00
/// <summary>
/// Resets count to default value.
/// </summary>
2016-10-13 22:13:20 +00:00
public virtual void ResetCount()
{
2018-10-16 02:40:51 +00:00
SetCountWithoutRolling(default);
2016-10-13 22:13:20 +00:00
}
2018-04-13 09:19:50 +00:00
2016-10-07 07:05:02 +00:00
/// <summary>
2016-10-07 21:14:35 +00:00
/// Calculates the duration of the roll-up animation by using the difference between the current visible value
/// and the new final value.
2016-10-07 07:05:02 +00:00
/// </summary>
/// <remarks>
/// To be used in conjunction with IsRollingProportional = true.
2016-10-07 21:14:35 +00:00
/// Unless a derived class needs to have a proportional rolling, it is not necessary to override this function.
2016-10-07 07:05:02 +00:00
/// </remarks>
/// <param name="currentValue">Current visible value.</param>
/// <param name="newValue">New final value.</param>
/// <returns>Calculated rollover duration in milliseconds.</returns>
2016-10-14 23:23:27 +00:00
protected virtual double GetProportionalDuration(T currentValue, T newValue)
2016-10-07 07:05:02 +00:00
{
return RollingDuration;
}
2018-04-13 09:19:50 +00:00
2016-10-07 07:05:02 +00:00
/// <summary>
/// Used to format counts.
/// </summary>
/// <param name="count">Count to format.</param>
2021-07-23 20:37:08 +00:00
/// <returns>Count formatted as a localisable string.</returns>
protected virtual LocalisableString FormatCount(T count)
2016-10-07 07:05:02 +00:00
{
return count.ToString();
}
2018-04-13 09:19:50 +00:00
2016-10-07 07:05:02 +00:00
/// <summary>
2016-10-07 21:14:35 +00:00
/// Called when the count is updated to add a transformer that changes the value of the visible count (i.e.
/// implement the rollover animation).
2016-10-07 07:05:02 +00:00
/// </summary>
/// <param name="currentValue">Count value before modification.</param>
/// <param name="newValue">Expected count value after modification.</param>
2016-10-14 23:23:27 +00:00
protected virtual void TransformCount(T currentValue, T newValue)
2016-10-07 07:05:02 +00:00
{
2016-10-14 03:54:02 +00:00
double rollingTotalDuration =
IsRollingProportional
2016-10-14 23:23:27 +00:00
? GetProportionalDuration(currentValue, newValue)
2016-10-14 03:54:02 +00:00
: RollingDuration;
2018-04-13 09:19:50 +00:00
2017-07-16 10:59:26 +00:00
this.TransformTo(nameof(DisplayedCount), newValue, rollingTotalDuration, RollingEasing);
2016-10-07 07:05:02 +00:00
}
2021-10-01 11:09:39 +00:00
/// <summary>
/// Creates the text. Delegates to <see cref="CreateSpriteText"/> by default.
/// </summary>
2021-09-30 08:54:52 +00:00
protected virtual IHasText CreateText() => CreateSpriteText();
2021-10-01 11:09:39 +00:00
/// <summary>
/// Creates an <see cref="OsuSpriteText"/> which may be used to display this counter's text.
/// May not be called if <see cref="CreateText"/> is overridden.
/// </summary>
protected virtual OsuSpriteText CreateSpriteText() => new OsuSpriteText
{
Font = OsuFont.Numeric.With(size: 40f),
};
2016-10-07 07:05:02 +00:00
}
}