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

233 lines
7.4 KiB
C#
Raw Normal View History

// Copyright (c) 2007-2017 ppy Pty Ltd <contact@ppy.sh>.
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
2016-10-13 22:13:20 +00:00
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;
2016-10-07 07:05:02 +00:00
using osu.Framework.Graphics.Transformations;
using System;
using System.Collections.Generic;
using System.Diagnostics;
using osu.Game.Graphics.Sprites;
2016-10-07 07:05:02 +00:00
namespace osu.Game.Graphics.UserInterface
{
2016-10-22 08:50:42 +00:00
public abstract class RollingCounter<T> : Container
2016-10-07 07:05:02 +00:00
{
/// <summary>
/// Type of the Transform to use.
/// </summary>
/// <remarks>
/// Must be a subclass of Transform<T>
/// </remarks>
2016-10-14 23:23:27 +00:00
protected virtual Type TransformType => typeof(Transform<T>);
2016-10-16 00:07:07 +00:00
protected SpriteText DisplayedCountSpriteText;
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;
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;
2016-10-07 07:05:02 +00:00
/// <summary>
/// Easing for the counter rollover animation.
/// </summary>
protected virtual EasingTypes RollingEasing => EasingTypes.Out;
2016-10-07 07:05:02 +00:00
2016-10-16 00:07:07 +00:00
private T displayedCount;
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
{
2016-10-16 00:07:07 +00:00
return displayedCount;
2016-10-07 07:05:02 +00:00
}
protected set
{
if (EqualityComparer<T>.Default.Equals(displayedCount, value))
2016-10-07 07:05:02 +00:00
return;
2016-10-16 00:07:07 +00:00
displayedCount = value;
DisplayedCountSpriteText.Text = FormatCount(value);
2016-10-07 07:05:02 +00:00
}
}
2017-02-07 07:15:45 +00:00
private T count;
2016-10-07 07:05:02 +00:00
/// <summary>
/// Actual value of counter.
/// </summary>
public virtual T Count
{
get
{
return count;
}
set
{
count = value;
2016-10-12 19:33:04 +00:00
if (IsLoaded)
2016-10-07 07:05:02 +00:00
{
2016-10-16 00:07:07 +00:00
TransformCount(displayedCount, count);
2016-10-07 07:05:02 +00:00
}
}
}
public void Set(T value)
{
Count = value;
}
public abstract void Increment(T amount);
2017-02-07 07:15:45 +00:00
private float textSize;
2016-10-13 22:13:20 +00:00
public float TextSize
{
get { return textSize; }
set
{
textSize = value;
2016-10-16 00:07:07 +00:00
DisplayedCountSpriteText.TextSize = value;
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-13 22:13:20 +00:00
Children = new Drawable[]
{
DisplayedCountSpriteText = new OsuSpriteText()
{
Font = @"Venera"
},
2016-10-13 22:13:20 +00:00
};
TextSize = 40;
2016-10-22 08:50:42 +00:00
AutoSizeAxes = Axes.Both;
2016-10-16 00:07:07 +00:00
DisplayedCount = Count;
2016-10-13 22:13:20 +00:00
2016-10-16 00:07:07 +00:00
DisplayedCountSpriteText.Text = FormatCount(count);
2016-11-08 23:13:20 +00:00
DisplayedCountSpriteText.Anchor = Anchor;
DisplayedCountSpriteText.Origin = Origin;
2016-10-07 07:05:02 +00:00
}
protected override void LoadComplete()
{
base.LoadComplete();
Flush(false, TransformType);
}
/// <summary>
/// Sets count value, bypassing rollover animation.
/// </summary>
/// <param name="count">New count value.</param>
public virtual void SetCountWithoutRolling(T count)
{
Count = count;
StopRolling();
}
/// <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()
{
2016-10-14 23:23:27 +00:00
Flush(false, TransformType);
2016-10-16 00:07:07 +00:00
DisplayedCount = Count;
}
/// <summary>
/// Resets count to default value.
/// </summary>
2016-10-13 22:13:20 +00:00
public virtual void ResetCount()
{
SetCountWithoutRolling(default(T));
}
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;
}
/// <summary>
/// Used to format counts.
/// </summary>
/// <param name="count">Count to format.</param>
/// <returns>Count formatted as a string.</returns>
2016-10-14 23:23:27 +00:00
protected virtual string FormatCount(T count)
2016-10-07 07:05:02 +00:00
{
return count.ToString();
}
/// <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
/// <seealso cref="TransformType"/>
protected virtual void TransformCount(T currentValue, T newValue)
{
2016-10-16 23:30:25 +00:00
Debug.Assert(
TransformType.IsSubclassOf(typeof(Transform<T>)) || TransformType == typeof(Transform<T>),
@"transformType should be a subclass of Transform<T>."
);
2016-10-28 08:42:00 +00:00
TransformCount((Transform<T>)Activator.CreateInstance(TransformType), currentValue, newValue);
}
2016-10-07 07:05:02 +00:00
/// <summary>
2016-10-14 23:23:27 +00:00
/// Intended to be used by TransformCount(T currentValue, T newValue).
2016-10-07 07:05:02 +00:00
/// </summary>
2016-10-14 23:23:27 +00:00
protected void TransformCount(Transform<T> transform, T currentValue, T newValue)
2016-10-07 07:05:02 +00:00
{
Type type = transform.GetType();
2016-10-14 23:23:27 +00:00
Flush(false, type);
2016-10-07 07:05:02 +00:00
2016-10-16 23:30:25 +00:00
if (RollingDuration < 1)
2016-10-07 07:05:02 +00:00
{
2016-10-16 00:07:07 +00:00
DisplayedCount = Count;
2016-10-07 07:05:02 +00:00
return;
}
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;
transform.StartTime = Time.Current;
transform.EndTime = Time.Current + rollingTotalDuration;
2016-10-07 07:05:02 +00:00
transform.StartValue = currentValue;
transform.EndValue = newValue;
transform.Easing = RollingEasing;
Transforms.Add(transform);
}
}
}