2022-05-31 23:17:39 +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.
|
|
|
|
|
2022-06-06 16:11:26 +08:00
|
|
|
using System;
|
2022-05-31 23:17:39 +08:00
|
|
|
using osu.Game.Rulesets.Difficulty.Preprocessing;
|
|
|
|
using osu.Game.Rulesets.Taiko.Difficulty.Preprocessing;
|
|
|
|
using osu.Game.Rulesets.Taiko.Objects;
|
|
|
|
|
|
|
|
namespace osu.Game.Rulesets.Taiko.Difficulty.Evaluators
|
|
|
|
{
|
|
|
|
public class StaminaEvaluator
|
|
|
|
{
|
|
|
|
/// <summary>
|
|
|
|
/// Applies a speed bonus dependent on the time since the last hit performed using this key.
|
|
|
|
/// </summary>
|
2022-06-22 17:17:19 +08:00
|
|
|
/// <param name="interval">The interval between the current and previous note hit using the same key.</param>
|
|
|
|
private static double speedBonus(double interval)
|
2022-05-31 23:17:39 +08:00
|
|
|
{
|
2022-07-16 19:20:25 +08:00
|
|
|
// Cap to 600bpm 1/4, 25ms note interval, 50ms key interval
|
2022-07-17 12:56:07 +08:00
|
|
|
// Interval will be capped at a very small value to avoid infinite/negative speed bonuses.
|
|
|
|
// TODO - This is a temporary measure as we need to implement methods of detecting playstyle-abuse of SpeedBonus.
|
2022-07-16 19:45:35 +08:00
|
|
|
interval = Math.Max(interval, 50);
|
2022-07-14 16:29:23 +08:00
|
|
|
|
2022-07-16 19:45:35 +08:00
|
|
|
return 30 / interval;
|
2022-05-31 23:17:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Evaluates the minimum mechanical stamina required to play the current object. This is calculated using the
|
2022-07-20 21:33:38 +08:00
|
|
|
/// maximum possible interval between two hits using the same key, by alternating 2 keys for each colour.
|
2022-05-31 23:17:39 +08:00
|
|
|
/// </summary>
|
|
|
|
public static double EvaluateDifficultyOf(DifficultyHitObject current)
|
|
|
|
{
|
2022-07-15 19:07:01 +08:00
|
|
|
if (current.BaseObject is not Hit)
|
2022-05-31 23:17:39 +08:00
|
|
|
{
|
|
|
|
return 0.0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find the previous hit object hit by the current key, which is two notes of the same colour prior.
|
|
|
|
TaikoDifficultyHitObject taikoCurrent = (TaikoDifficultyHitObject)current;
|
2022-07-15 19:07:01 +08:00
|
|
|
TaikoDifficultyHitObject? keyPrevious = taikoCurrent.PreviousMono(1);
|
2022-06-09 17:22:55 +08:00
|
|
|
|
2022-06-01 05:20:08 +08:00
|
|
|
if (keyPrevious == null)
|
2022-05-31 23:17:39 +08:00
|
|
|
{
|
2022-06-01 05:20:08 +08:00
|
|
|
// There is no previous hit object hit by the current key
|
|
|
|
return 0.0;
|
|
|
|
}
|
2022-05-31 23:17:39 +08:00
|
|
|
|
2022-07-14 16:29:23 +08:00
|
|
|
double objectStrain = 0.5; // Add a base strain to all objects
|
|
|
|
objectStrain += speedBonus(taikoCurrent.StartTime - keyPrevious.StartTime);
|
2022-05-31 23:17:39 +08:00
|
|
|
return objectStrain;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|