2016-10-12 07:11:40 +00:00
|
|
|
|
using osu.Framework;
|
|
|
|
|
using osu.Framework.Graphics;
|
2016-10-07 07:05:02 +00:00
|
|
|
|
using osu.Framework.Graphics.Containers;
|
|
|
|
|
using osu.Framework.Graphics.Transformations;
|
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
2016-10-07 21:59:52 +00:00
|
|
|
|
using System.Diagnostics;
|
2016-10-07 07:05:02 +00:00
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Text;
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
|
|
|
|
|
namespace osu.Game.Graphics.UserInterface
|
|
|
|
|
{
|
|
|
|
|
/// <summary>
|
2016-10-09 00:11:01 +00:00
|
|
|
|
/// Skeleton for a counter which value rolls-up in a lapse of time.
|
2016-10-07 07:05:02 +00:00
|
|
|
|
/// </summary>
|
2016-10-09 00:11:01 +00:00
|
|
|
|
/// <remarks>
|
|
|
|
|
/// This class only abstracts the basics to roll-up a value in a lapse of time by using Transforms.
|
|
|
|
|
/// In order to show a value, you must implement a way to display it, i.e., as a numeric counter or a bar.
|
|
|
|
|
/// </remarks>
|
2016-10-07 07:05:02 +00:00
|
|
|
|
/// <typeparam name="T">Type of the actual counter.</typeparam>
|
2016-10-12 19:33:04 +00:00
|
|
|
|
public abstract class RollingCounter<T> : AutoSizeContainer
|
2016-10-07 07:05:02 +00:00
|
|
|
|
{
|
2016-10-07 21:59:52 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Type of the Transform to use.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <remarks>
|
|
|
|
|
/// Must be a subclass of Transform<T>
|
|
|
|
|
/// </remarks>
|
|
|
|
|
protected virtual Type transformType => typeof(Transform<T>);
|
|
|
|
|
|
2016-10-07 07:05:02 +00:00
|
|
|
|
protected ulong RollingTotalDuration = 0;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// If true, each time the Count is updated, it will roll over from the current visible value.
|
2016-10-07 21:14:35 +00:00
|
|
|
|
/// Else, it will roll up from the current count value.
|
2016-10-07 07:05:02 +00:00
|
|
|
|
/// </summary>
|
|
|
|
|
public bool IsRollingContinuous = true;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
2016-10-07 21:14:35 +00:00
|
|
|
|
/// If true, the roll-up duration will be proportional to the counter.
|
2016-10-07 07:05:02 +00:00
|
|
|
|
/// </summary>
|
|
|
|
|
public bool IsRollingProportional = false;
|
|
|
|
|
|
|
|
|
|
/// <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>
|
|
|
|
|
public ulong RollingDuration = 0;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Easing for the counter rollover animation.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public EasingTypes RollingEasing = EasingTypes.None;
|
|
|
|
|
|
2016-10-09 19:02:44 +00:00
|
|
|
|
protected T prevVisibleCount;
|
2016-10-07 07:05:02 +00:00
|
|
|
|
protected T visibleCount;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Value shown at the current moment.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public virtual T VisibleCount
|
|
|
|
|
{
|
|
|
|
|
get
|
|
|
|
|
{
|
|
|
|
|
return visibleCount;
|
|
|
|
|
}
|
|
|
|
|
protected set
|
|
|
|
|
{
|
2016-10-09 19:02:44 +00:00
|
|
|
|
prevVisibleCount = visibleCount;
|
2016-10-07 07:05:02 +00:00
|
|
|
|
if (visibleCount.Equals(value))
|
|
|
|
|
return;
|
|
|
|
|
visibleCount = value;
|
2016-10-09 19:02:44 +00:00
|
|
|
|
transformVisibleCount(prevVisibleCount, value);
|
2016-10-07 07:05:02 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-10-09 19:02:44 +00:00
|
|
|
|
protected T prevCount;
|
2016-10-07 07:05:02 +00:00
|
|
|
|
protected T count;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Actual value of counter.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public virtual T Count
|
|
|
|
|
{
|
|
|
|
|
get
|
|
|
|
|
{
|
|
|
|
|
return count;
|
|
|
|
|
}
|
|
|
|
|
set
|
|
|
|
|
{
|
2016-10-09 19:02:44 +00:00
|
|
|
|
prevCount = count;
|
|
|
|
|
count = value;
|
2016-10-12 19:33:04 +00:00
|
|
|
|
if (IsLoaded)
|
2016-10-07 07:05:02 +00:00
|
|
|
|
{
|
2016-10-07 21:14:35 +00:00
|
|
|
|
RollingTotalDuration =
|
|
|
|
|
IsRollingProportional
|
2016-10-07 21:59:52 +00:00
|
|
|
|
? getProportionalDuration(VisibleCount, value)
|
2016-10-07 21:14:35 +00:00
|
|
|
|
: RollingDuration;
|
2016-10-09 20:19:35 +00:00
|
|
|
|
transformCount(IsRollingContinuous ? VisibleCount : prevCount, value);
|
2016-10-07 07:05:02 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-10-12 19:33:04 +00:00
|
|
|
|
protected RollingCounter() : base()
|
2016-10-07 21:59:52 +00:00
|
|
|
|
{
|
2016-10-09 19:48:24 +00:00
|
|
|
|
Debug.Assert(
|
|
|
|
|
transformType.IsSubclassOf(typeof(Transform<T>)) || transformType == typeof(Transform<T>),
|
|
|
|
|
@"transformType should be a subclass of Transform<T>."
|
|
|
|
|
);
|
2016-10-07 21:59:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-10-12 07:11:40 +00:00
|
|
|
|
public override void Load(BaseGame game)
|
2016-10-07 07:05:02 +00:00
|
|
|
|
{
|
2016-10-12 07:11:40 +00:00
|
|
|
|
base.Load(game);
|
2016-10-12 19:33:04 +00:00
|
|
|
|
|
2016-10-07 21:59:52 +00:00
|
|
|
|
removeTransforms(transformType);
|
2016-10-12 19:33:04 +00:00
|
|
|
|
|
2016-10-07 07:05:02 +00:00
|
|
|
|
if (Count == null)
|
|
|
|
|
ResetCount();
|
|
|
|
|
VisibleCount = Count;
|
|
|
|
|
}
|
|
|
|
|
|
2016-10-07 21:59:52 +00:00
|
|
|
|
/// <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>
|
|
|
|
|
/// Stops rollover animation, forcing the visible count to be the actual count.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public virtual void StopRolling()
|
|
|
|
|
{
|
|
|
|
|
removeTransforms(transformType);
|
|
|
|
|
VisibleCount = Count;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Resets count to default value.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public abstract void ResetCount();
|
|
|
|
|
|
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>
|
2016-10-07 21:59:52 +00:00
|
|
|
|
/// 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-07 21:59:52 +00:00
|
|
|
|
protected virtual ulong 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>
|
|
|
|
|
protected virtual string formatCount(T count)
|
|
|
|
|
{
|
|
|
|
|
return count.ToString();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected void updateTransforms(Type type)
|
|
|
|
|
{
|
|
|
|
|
foreach (ITransform t in Transforms.AliveItems)
|
2016-10-07 21:59:52 +00:00
|
|
|
|
if (t.GetType() == type)
|
2016-10-07 07:05:02 +00:00
|
|
|
|
t.Apply(this);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected void removeTransforms(Type type)
|
|
|
|
|
{
|
2016-10-07 21:59:52 +00:00
|
|
|
|
Transforms.RemoveAll(t => t.GetType() == type);
|
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>
|
|
|
|
|
/// <remarks>
|
2016-10-07 21:14:35 +00:00
|
|
|
|
/// Unless you need to set a custom animation according to the current or new value of the count, the
|
|
|
|
|
/// recommended approach is to call transformCount(CustomTransformer(Clock), currentValue, newValue), where
|
2016-10-07 21:59:52 +00:00
|
|
|
|
/// CustomTransformer is of type transformerType.
|
2016-10-07 21:14:35 +00:00
|
|
|
|
/// By using this approach, there is no need to check if the Clock is not null; this validation is done before
|
|
|
|
|
/// adding the transformer.
|
2016-10-07 07:05:02 +00:00
|
|
|
|
/// </remarks>
|
2016-10-07 21:59:52 +00:00
|
|
|
|
/// <seealso cref="transformType"/>
|
|
|
|
|
protected virtual void transformCount(T currentValue, T newValue)
|
|
|
|
|
{
|
|
|
|
|
object[] parameters = { Clock };
|
|
|
|
|
transformCount((Transform<T>)Activator.CreateInstance(transformType, parameters), currentValue, newValue);
|
|
|
|
|
}
|
2016-10-07 07:05:02 +00:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Intended to be used by transformCount().
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <see cref="transformCount"/>
|
|
|
|
|
protected void transformCount(Transform<T> transform, T currentValue, T newValue)
|
|
|
|
|
{
|
|
|
|
|
Type type = transform.GetType();
|
|
|
|
|
|
|
|
|
|
updateTransforms(type);
|
|
|
|
|
removeTransforms(type);
|
|
|
|
|
|
|
|
|
|
if (Clock == null)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (RollingDuration == 0)
|
|
|
|
|
{
|
|
|
|
|
VisibleCount = Count;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
transform.StartTime = Time;
|
|
|
|
|
transform.EndTime = Time + RollingTotalDuration;
|
|
|
|
|
transform.StartValue = currentValue;
|
|
|
|
|
transform.EndValue = newValue;
|
|
|
|
|
transform.Easing = RollingEasing;
|
|
|
|
|
|
|
|
|
|
Transforms.Add(transform);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// This procedure is called each time the visible count value is updated.
|
|
|
|
|
/// Override to create custom animations.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="currentValue">Visible count value before modification.</param>
|
|
|
|
|
/// <param name="newValue">Expected visible count value after modification-</param>
|
2016-10-09 00:11:01 +00:00
|
|
|
|
protected abstract void transformVisibleCount(T currentValue, T newValue);
|
2016-10-07 07:05:02 +00:00
|
|
|
|
}
|
|
|
|
|
}
|