2021-04-03 17:52:36 +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.
|
|
|
|
|
|
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using osu.Game.Rulesets.Difficulty.Preprocessing;
|
|
|
|
|
using osu.Game.Rulesets.Mods;
|
|
|
|
|
|
|
|
|
|
namespace osu.Game.Rulesets.Difficulty.Skills
|
|
|
|
|
{
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Used to processes strain values of <see cref="DifficultyHitObject"/>s, keep track of strain levels caused by the processed objects
|
|
|
|
|
/// and to calculate a final difficulty value representing the difficulty of hitting all the processed objects.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public abstract class StrainSkill : Skill
|
|
|
|
|
{
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// The weight by which each strain value decays.
|
|
|
|
|
/// </summary>
|
|
|
|
|
protected virtual double DecayWeight => 0.9;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// The length of each strain section.
|
|
|
|
|
/// </summary>
|
|
|
|
|
protected virtual int SectionLength => 400;
|
|
|
|
|
|
2021-10-07 16:30:18 +08:00
|
|
|
|
private double currentSectionPeak; // We also keep track of the peak strain level in the current section.
|
2021-04-03 17:52:36 +08:00
|
|
|
|
private double currentSectionEnd;
|
|
|
|
|
|
|
|
|
|
private readonly List<double> strainPeaks = new List<double>();
|
2024-11-07 17:53:53 +08:00
|
|
|
|
protected List<double> ObjectStrains = new List<double>(); // Store individual strains
|
2021-04-03 17:52:36 +08:00
|
|
|
|
|
|
|
|
|
protected StrainSkill(Mod[] mods)
|
|
|
|
|
: base(mods)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-17 06:14:29 +08:00
|
|
|
|
/// <summary>
|
2021-08-17 06:36:14 +08:00
|
|
|
|
/// Returns the strain value at <see cref="DifficultyHitObject"/>. This value is calculated with or without respect to previous objects.
|
2021-08-17 06:14:29 +08:00
|
|
|
|
/// </summary>
|
|
|
|
|
protected abstract double StrainValueAt(DifficultyHitObject current);
|
|
|
|
|
|
2021-04-03 17:52:36 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Process a <see cref="DifficultyHitObject"/> and update current strain values accordingly.
|
|
|
|
|
/// </summary>
|
2022-05-23 04:45:27 +08:00
|
|
|
|
public sealed override void Process(DifficultyHitObject current)
|
2021-04-03 17:52:36 +08:00
|
|
|
|
{
|
|
|
|
|
// The first object doesn't generate a strain, so we begin with an incremented section end
|
2022-06-09 17:49:11 +08:00
|
|
|
|
if (current.Index == 0)
|
2021-04-03 17:52:36 +08:00
|
|
|
|
currentSectionEnd = Math.Ceiling(current.StartTime / SectionLength) * SectionLength;
|
|
|
|
|
|
|
|
|
|
while (current.StartTime > currentSectionEnd)
|
|
|
|
|
{
|
|
|
|
|
saveCurrentPeak();
|
2022-05-22 23:26:22 +08:00
|
|
|
|
startNewSectionFrom(currentSectionEnd, current);
|
2021-04-03 17:52:36 +08:00
|
|
|
|
currentSectionEnd += SectionLength;
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-07 17:53:53 +08:00
|
|
|
|
double strain = StrainValueAt(current);
|
|
|
|
|
currentSectionPeak = Math.Max(strain, currentSectionPeak);
|
|
|
|
|
|
|
|
|
|
// Store the strain value for the object
|
|
|
|
|
ObjectStrains.Add(strain);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Calculates the number of strains weighted against the top strain.
|
|
|
|
|
/// The result is scaled by clock rate as it affects the total number of strains.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public double CountDifficultStrains()
|
|
|
|
|
{
|
|
|
|
|
if (ObjectStrains.Count == 0)
|
|
|
|
|
return 0.0;
|
|
|
|
|
|
|
|
|
|
double consistentTopStrain = DifficultyValue() / 10; // What would the top strain be if all strain values were identical
|
|
|
|
|
// Use a weighted sum of all strains. Constants are arbitrary and give nice values
|
|
|
|
|
return ObjectStrains.Sum(s => 1.1 / (1 + Math.Exp(-10 * (s / consistentTopStrain - 0.88))));
|
2021-04-03 17:52:36 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Saves the current peak strain level to the list of strain peaks, which will be used to calculate an overall difficulty.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private void saveCurrentPeak()
|
|
|
|
|
{
|
|
|
|
|
strainPeaks.Add(currentSectionPeak);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Sets the initial strain level for a new section.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="time">The beginning of the new section in milliseconds.</param>
|
2022-05-22 23:26:22 +08:00
|
|
|
|
/// <param name="current">The current hit object.</param>
|
|
|
|
|
private void startNewSectionFrom(double time, DifficultyHitObject current)
|
2021-04-03 17:52:36 +08:00
|
|
|
|
{
|
2021-08-17 06:14:29 +08:00
|
|
|
|
// The maximum strain of the new section is not zero by default
|
2021-04-03 17:52:36 +08:00
|
|
|
|
// This means we need to capture the strain level at the beginning of the new section, and use that as the initial peak level.
|
2022-05-22 23:26:22 +08:00
|
|
|
|
currentSectionPeak = CalculateInitialStrain(time, current);
|
2021-04-03 17:52:36 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Retrieves the peak strain at a point in time.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="time">The time to retrieve the peak strain at.</param>
|
2022-05-22 23:26:22 +08:00
|
|
|
|
/// <param name="current">The current hit object.</param>
|
2021-04-03 17:52:36 +08:00
|
|
|
|
/// <returns>The peak strain.</returns>
|
2022-05-22 23:26:22 +08:00
|
|
|
|
protected abstract double CalculateInitialStrain(double time, DifficultyHitObject current);
|
2021-04-03 17:52:36 +08:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Returns a live enumerable of the peak strains for each <see cref="SectionLength"/> section of the beatmap,
|
|
|
|
|
/// including the peak of the current section.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public IEnumerable<double> GetCurrentStrainPeaks() => strainPeaks.Append(currentSectionPeak);
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Returns the calculated difficulty value representing all <see cref="DifficultyHitObject"/>s that have been processed up to this point.
|
|
|
|
|
/// </summary>
|
2021-06-16 09:34:46 +08:00
|
|
|
|
public override double DifficultyValue()
|
2021-04-03 17:52:36 +08:00
|
|
|
|
{
|
|
|
|
|
double difficulty = 0;
|
|
|
|
|
double weight = 1;
|
|
|
|
|
|
2022-05-03 15:06:20 +08:00
|
|
|
|
// Sections with 0 strain are excluded to avoid worst-case time complexity of the following sort (e.g. /b/2351871).
|
|
|
|
|
// These sections will not contribute to the difficulty.
|
|
|
|
|
var peaks = GetCurrentStrainPeaks().Where(p => p > 0);
|
|
|
|
|
|
2021-04-03 17:52:36 +08:00
|
|
|
|
// Difficulty is the weighted sum of the highest strains from every section.
|
|
|
|
|
// We're sorting from highest to lowest strain.
|
2024-02-09 01:01:00 +08:00
|
|
|
|
foreach (double strain in peaks.OrderDescending())
|
2021-04-03 17:52:36 +08:00
|
|
|
|
{
|
|
|
|
|
difficulty += strain * weight;
|
|
|
|
|
weight *= DecayWeight;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return difficulty;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|