1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-19 19:52:58 +08:00
osu-lazer/osu.Game.Rulesets.Osu/Difficulty/Skills/OsuStrainSkill.cs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

63 lines
2.4 KiB
C#
Raw Normal View History

2021-06-16 09:34:46 +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;
2021-06-13 21:18:35 +08:00
using System.Collections.Generic;
using osu.Game.Rulesets.Difficulty.Skills;
using osu.Game.Rulesets.Mods;
using System.Linq;
2021-06-17 01:54:22 +08:00
using osu.Framework.Utils;
2021-06-13 21:18:35 +08:00
namespace osu.Game.Rulesets.Osu.Difficulty.Skills
{
2021-08-17 21:39:18 +08:00
public abstract class OsuStrainSkill : StrainSkill
2021-06-13 21:18:35 +08:00
{
2021-06-18 03:41:06 +08:00
/// <summary>
/// The number of sections with the highest strains, which the peak strain reductions will apply to.
/// This is done in order to decrease their impact on the overall difficulty of the map for this skill.
/// </summary>
2021-06-15 01:22:35 +08:00
protected virtual int ReducedSectionCount => 10;
2021-06-18 03:41:06 +08:00
/// <summary>
/// The baseline multiplier applied to the section with the biggest strain.
/// </summary>
2021-06-16 21:13:46 +08:00
protected virtual double ReducedStrainBaseline => 0.75;
2021-06-18 03:41:06 +08:00
protected OsuStrainSkill(Mod[] mods)
2021-06-16 09:34:46 +08:00
: base(mods)
2021-06-13 21:18:35 +08:00
{
}
2021-06-16 09:34:46 +08:00
public override double DifficultyValue()
2021-06-13 21:18:35 +08:00
{
double difficulty = 0;
double weight = 1;
// 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);
List<double> strains = peaks.OrderDescending().ToList();
2021-06-13 21:18:35 +08:00
2021-06-16 09:34:46 +08:00
// We are reducing the highest strains first to account for extreme difficulty spikes
for (int i = 0; i < Math.Min(strains.Count, ReducedSectionCount); i++)
2021-06-13 21:18:35 +08:00
{
2021-06-18 03:41:06 +08:00
double scale = Math.Log10(Interpolation.Lerp(1, 10, Math.Clamp((float)i / ReducedSectionCount, 0, 1)));
strains[i] *= Interpolation.Lerp(ReducedStrainBaseline, 1.0, scale);
2021-06-13 21:18:35 +08:00
}
// Difficulty is the weighted sum of the highest strains from every section.
// We're sorting from highest to lowest strain.
foreach (double strain in strains.OrderDescending())
2021-06-13 21:18:35 +08:00
{
difficulty += strain * weight;
weight *= DecayWeight;
}
2024-08-05 20:57:02 +08:00
return difficulty;
2021-06-13 21:18:35 +08:00
}
public static double DifficultyToPerformance(double difficulty) => Math.Pow(5.0 * Math.Max(1.0, difficulty / 0.0675) - 4.0, 3.0) / 100000.0;
2021-06-13 21:18:35 +08:00
}
}