mirror of
https://github.com/osukey/osukey.git
synced 2025-05-23 06:27:24 +09:00
Stamina Skill Rewrite and Refactor
This commit is contained in:
parent
6f7c675dfe
commit
48b7e0f090
@ -1,16 +1,50 @@
|
|||||||
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
// 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.
|
// See the LICENCE file in the repository root for full licence text.
|
||||||
|
|
||||||
using System.Linq;
|
using System;
|
||||||
using osu.Game.Rulesets.Difficulty.Preprocessing;
|
using osu.Game.Rulesets.Difficulty.Preprocessing;
|
||||||
using osu.Game.Rulesets.Difficulty.Skills;
|
using osu.Game.Rulesets.Difficulty.Skills;
|
||||||
using osu.Game.Rulesets.Difficulty.Utils;
|
|
||||||
using osu.Game.Rulesets.Mods;
|
using osu.Game.Rulesets.Mods;
|
||||||
using osu.Game.Rulesets.Taiko.Difficulty.Preprocessing;
|
using osu.Game.Rulesets.Taiko.Difficulty.Preprocessing;
|
||||||
using osu.Game.Rulesets.Taiko.Objects;
|
using osu.Game.Rulesets.Taiko.Objects;
|
||||||
|
|
||||||
namespace osu.Game.Rulesets.Taiko.Difficulty.Skills
|
namespace osu.Game.Rulesets.Taiko.Difficulty.Skills
|
||||||
{
|
{
|
||||||
|
class SingleKeyStamina
|
||||||
|
{
|
||||||
|
private double previousHitTime = -1;
|
||||||
|
|
||||||
|
private double strainValueOf(DifficultyHitObject current)
|
||||||
|
{
|
||||||
|
if (previousHitTime == -1)
|
||||||
|
{
|
||||||
|
previousHitTime = current.StartTime;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
double objectStrain = 0.5;
|
||||||
|
objectStrain += speedBonus(current.StartTime - previousHitTime);
|
||||||
|
previousHitTime = current.StartTime;
|
||||||
|
return objectStrain;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public double StrainValueAt(DifficultyHitObject current)
|
||||||
|
{
|
||||||
|
return strainValueOf(current);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Applies a speed bonus dependent on the time since the last hit performed using this key.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="notePairDuration">The duration between the current and previous note hit using the same key.</param>
|
||||||
|
private double speedBonus(double notePairDuration)
|
||||||
|
{
|
||||||
|
return 175 / Math.Pow(notePairDuration + 100, 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Calculates the stamina coefficient of taiko difficulty.
|
/// Calculates the stamina coefficient of taiko difficulty.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@ -22,39 +56,43 @@ namespace osu.Game.Rulesets.Taiko.Difficulty.Skills
|
|||||||
protected override double SkillMultiplier => 1;
|
protected override double SkillMultiplier => 1;
|
||||||
protected override double StrainDecayBase => 0.4;
|
protected override double StrainDecayBase => 0.4;
|
||||||
|
|
||||||
/// <summary>
|
private SingleKeyStamina[] keyStamina = new SingleKeyStamina[4]
|
||||||
/// Maximum number of entries to keep in <see cref="notePairDurationHistory"/>.
|
{
|
||||||
/// </summary>
|
new SingleKeyStamina(),
|
||||||
private const int max_history_length = 2;
|
new SingleKeyStamina(),
|
||||||
|
new SingleKeyStamina(),
|
||||||
|
new SingleKeyStamina()
|
||||||
|
};
|
||||||
|
|
||||||
/// <summary>
|
private int donIndex = 1;
|
||||||
/// The index of the hand this <see cref="Stamina"/> instance is associated with.
|
private int katIndex = 3;
|
||||||
/// </summary>
|
|
||||||
/// <remarks>
|
|
||||||
/// The value of 0 indicates the left hand (full alternating gameplay starting with left hand is assumed).
|
|
||||||
/// This naturally translates onto index offsets of the objects in the map.
|
|
||||||
/// </remarks>
|
|
||||||
private readonly int hand;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Stores the last <see cref="max_history_length"/> durations between notes hit with the hand indicated by <see cref="hand"/>.
|
|
||||||
/// </summary>
|
|
||||||
private readonly LimitedCapacityQueue<double> notePairDurationHistory = new LimitedCapacityQueue<double>(max_history_length);
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Stores the <see cref="DifficultyHitObject.DeltaTime"/> of the last object that was hit by the <i>other</i> hand.
|
|
||||||
/// </summary>
|
|
||||||
private double offhandObjectDuration = double.MaxValue;
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Creates a <see cref="Stamina"/> skill.
|
/// Creates a <see cref="Stamina"/> skill.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="mods">Mods for use in skill calculations.</param>
|
/// <param name="mods">Mods for use in skill calculations.</param>
|
||||||
/// <param name="rightHand">Whether this instance is performing calculations for the right hand.</param>
|
public Stamina(Mod[] mods)
|
||||||
public Stamina(Mod[] mods, bool rightHand)
|
|
||||||
: base(mods)
|
: base(mods)
|
||||||
{
|
{
|
||||||
hand = rightHand ? 1 : 0;
|
}
|
||||||
|
|
||||||
|
private SingleKeyStamina getNextSingleKeyStamina(TaikoDifficultyHitObject current)
|
||||||
|
{
|
||||||
|
if (current.HitType == HitType.Centre)
|
||||||
|
{
|
||||||
|
donIndex = donIndex == 0 ? 1 : 0;
|
||||||
|
return keyStamina[donIndex];
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
katIndex = katIndex == 2 ? 3 : 2;
|
||||||
|
return keyStamina[katIndex];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private double sigmoid(double val, double center, double width)
|
||||||
|
{
|
||||||
|
return Math.Tanh(Math.E * -(val - center) / width);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override double StrainValueOf(DifficultyHitObject current)
|
protected override double StrainValueOf(DifficultyHitObject current)
|
||||||
@ -65,52 +103,9 @@ namespace osu.Game.Rulesets.Taiko.Difficulty.Skills
|
|||||||
}
|
}
|
||||||
|
|
||||||
TaikoDifficultyHitObject hitObject = (TaikoDifficultyHitObject)current;
|
TaikoDifficultyHitObject hitObject = (TaikoDifficultyHitObject)current;
|
||||||
|
double objectStrain = getNextSingleKeyStamina(hitObject).StrainValueAt(hitObject);
|
||||||
|
|
||||||
if (hitObject.ObjectIndex % 2 == hand)
|
return objectStrain;
|
||||||
{
|
|
||||||
double objectStrain = 1;
|
|
||||||
|
|
||||||
if (hitObject.ObjectIndex == 1)
|
|
||||||
return 1;
|
|
||||||
|
|
||||||
notePairDurationHistory.Enqueue(hitObject.DeltaTime + offhandObjectDuration);
|
|
||||||
|
|
||||||
double shortestRecentNote = notePairDurationHistory.Min();
|
|
||||||
objectStrain += speedBonus(shortestRecentNote);
|
|
||||||
|
|
||||||
if (hitObject.StaminaCheese)
|
|
||||||
objectStrain *= cheesePenalty(hitObject.DeltaTime + offhandObjectDuration);
|
|
||||||
|
|
||||||
return objectStrain;
|
|
||||||
}
|
|
||||||
|
|
||||||
offhandObjectDuration = hitObject.DeltaTime;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Applies a penalty for hit objects marked with <see cref="TaikoDifficultyHitObject.StaminaCheese"/>.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="notePairDuration">The duration between the current and previous note hit using the hand indicated by <see cref="hand"/>.</param>
|
|
||||||
private double cheesePenalty(double notePairDuration)
|
|
||||||
{
|
|
||||||
if (notePairDuration > 125) return 1;
|
|
||||||
if (notePairDuration < 100) return 0.6;
|
|
||||||
|
|
||||||
return 0.6 + (notePairDuration - 100) * 0.016;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Applies a speed bonus dependent on the time since the last hit performed using this hand.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="notePairDuration">The duration between the current and previous note hit using the hand indicated by <see cref="hand"/>.</param>
|
|
||||||
private double speedBonus(double notePairDuration)
|
|
||||||
{
|
|
||||||
if (notePairDuration >= 200) return 0;
|
|
||||||
|
|
||||||
double bonus = 200 - notePairDuration;
|
|
||||||
bonus *= bonus;
|
|
||||||
return bonus / 100000;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -22,7 +22,7 @@ namespace osu.Game.Rulesets.Taiko.Difficulty
|
|||||||
{
|
{
|
||||||
private const double rhythm_skill_multiplier = 0.014;
|
private const double rhythm_skill_multiplier = 0.014;
|
||||||
private const double colour_skill_multiplier = 0.01;
|
private const double colour_skill_multiplier = 0.01;
|
||||||
private const double stamina_skill_multiplier = 0.02;
|
private const double stamina_skill_multiplier = 0.021;
|
||||||
|
|
||||||
public TaikoDifficultyCalculator(IRulesetInfo ruleset, IWorkingBeatmap beatmap)
|
public TaikoDifficultyCalculator(IRulesetInfo ruleset, IWorkingBeatmap beatmap)
|
||||||
: base(ruleset, beatmap)
|
: base(ruleset, beatmap)
|
||||||
@ -33,8 +33,7 @@ namespace osu.Game.Rulesets.Taiko.Difficulty
|
|||||||
{
|
{
|
||||||
new Colour(mods),
|
new Colour(mods),
|
||||||
new Rhythm(mods),
|
new Rhythm(mods),
|
||||||
new Stamina(mods, true),
|
new Stamina(mods)
|
||||||
new Stamina(mods, false),
|
|
||||||
};
|
};
|
||||||
|
|
||||||
protected override Mod[] DifficultyAdjustmentMods => new Mod[]
|
protected override Mod[] DifficultyAdjustmentMods => new Mod[]
|
||||||
@ -58,7 +57,6 @@ namespace osu.Game.Rulesets.Taiko.Difficulty
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
new StaminaCheeseDetector(taikoDifficultyHitObjects).FindCheese();
|
|
||||||
return taikoDifficultyHitObjects;
|
return taikoDifficultyHitObjects;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -69,17 +67,16 @@ namespace osu.Game.Rulesets.Taiko.Difficulty
|
|||||||
|
|
||||||
var colour = (Colour)skills[0];
|
var colour = (Colour)skills[0];
|
||||||
var rhythm = (Rhythm)skills[1];
|
var rhythm = (Rhythm)skills[1];
|
||||||
var staminaRight = (Stamina)skills[2];
|
var stamina = (Stamina)skills[2];
|
||||||
var staminaLeft = (Stamina)skills[3];
|
|
||||||
|
|
||||||
double colourRating = colour.DifficultyValue() * colour_skill_multiplier;
|
double colourRating = colour.DifficultyValue() * colour_skill_multiplier;
|
||||||
double rhythmRating = rhythm.DifficultyValue() * rhythm_skill_multiplier;
|
double rhythmRating = rhythm.DifficultyValue() * rhythm_skill_multiplier;
|
||||||
double staminaRating = (staminaRight.DifficultyValue() + staminaLeft.DifficultyValue()) * stamina_skill_multiplier;
|
double staminaRating = stamina.DifficultyValue() * stamina_skill_multiplier;
|
||||||
|
|
||||||
double staminaPenalty = simpleColourPenalty(staminaRating, colourRating);
|
double staminaPenalty = simpleColourPenalty(staminaRating, colourRating);
|
||||||
staminaRating *= staminaPenalty;
|
staminaRating *= staminaPenalty;
|
||||||
|
|
||||||
double combinedRating = locallyCombinedDifficulty(colour, rhythm, staminaRight, staminaLeft, staminaPenalty);
|
double combinedRating = locallyCombinedDifficulty(colour, rhythm, stamina, staminaPenalty);
|
||||||
double separatedRating = norm(1.5, colourRating, rhythmRating, staminaRating);
|
double separatedRating = norm(1.5, colourRating, rhythmRating, staminaRating);
|
||||||
double starRating = 1.4 * separatedRating + 0.5 * combinedRating;
|
double starRating = 1.4 * separatedRating + 0.5 * combinedRating;
|
||||||
starRating = rescale(starRating);
|
starRating = rescale(starRating);
|
||||||
@ -127,20 +124,19 @@ namespace osu.Game.Rulesets.Taiko.Difficulty
|
|||||||
/// For each section, the peak strains of all separate skills are combined into a single peak strain for the section.
|
/// For each section, the peak strains of all separate skills are combined into a single peak strain for the section.
|
||||||
/// The resulting partial rating of the beatmap is a weighted sum of the combined peaks (higher peaks are weighted more).
|
/// The resulting partial rating of the beatmap is a weighted sum of the combined peaks (higher peaks are weighted more).
|
||||||
/// </remarks>
|
/// </remarks>
|
||||||
private double locallyCombinedDifficulty(Colour colour, Rhythm rhythm, Stamina staminaRight, Stamina staminaLeft, double staminaPenalty)
|
private double locallyCombinedDifficulty(Colour colour, Rhythm rhythm, Stamina stamina, double staminaPenalty)
|
||||||
{
|
{
|
||||||
List<double> peaks = new List<double>();
|
List<double> peaks = new List<double>();
|
||||||
|
|
||||||
var colourPeaks = colour.GetCurrentStrainPeaks().ToList();
|
var colourPeaks = colour.GetCurrentStrainPeaks().ToList();
|
||||||
var rhythmPeaks = rhythm.GetCurrentStrainPeaks().ToList();
|
var rhythmPeaks = rhythm.GetCurrentStrainPeaks().ToList();
|
||||||
var staminaRightPeaks = staminaRight.GetCurrentStrainPeaks().ToList();
|
var staminaPeaks = stamina.GetCurrentStrainPeaks().ToList();
|
||||||
var staminaLeftPeaks = staminaLeft.GetCurrentStrainPeaks().ToList();
|
|
||||||
|
|
||||||
for (int i = 0; i < colourPeaks.Count; i++)
|
for (int i = 0; i < colourPeaks.Count; i++)
|
||||||
{
|
{
|
||||||
double colourPeak = colourPeaks[i] * colour_skill_multiplier;
|
double colourPeak = colourPeaks[i] * colour_skill_multiplier;
|
||||||
double rhythmPeak = rhythmPeaks[i] * rhythm_skill_multiplier;
|
double rhythmPeak = rhythmPeaks[i] * rhythm_skill_multiplier;
|
||||||
double staminaPeak = (staminaRightPeaks[i] + staminaLeftPeaks[i]) * stamina_skill_multiplier * staminaPenalty;
|
double staminaPeak = staminaPeaks[i] * stamina_skill_multiplier * staminaPenalty;
|
||||||
|
|
||||||
double peak = norm(2, colourPeak, rhythmPeak, staminaPeak);
|
double peak = norm(2, colourPeak, rhythmPeak, staminaPeak);
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user