mirror of
https://github.com/ppy/osu
synced 2025-01-23 22:33:05 +00:00
112 lines
5.0 KiB
C#
112 lines
5.0 KiB
C#
// 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.
|
|
|
|
namespace osu.Game.Beatmaps
|
|
{
|
|
/// <summary>
|
|
/// A representation of all top-level difficulty settings for a beatmap.
|
|
/// </summary>
|
|
public interface IBeatmapDifficultyInfo
|
|
{
|
|
/// <summary>
|
|
/// The default value used for all difficulty settings except <see cref="SliderMultiplier"/> and <see cref="SliderTickRate"/>.
|
|
/// </summary>
|
|
const float DEFAULT_DIFFICULTY = 5;
|
|
|
|
/// <summary>
|
|
/// The drain rate of the associated beatmap.
|
|
/// </summary>
|
|
float DrainRate { get; }
|
|
|
|
/// <summary>
|
|
/// The circle size of the associated beatmap.
|
|
/// </summary>
|
|
float CircleSize { get; }
|
|
|
|
/// <summary>
|
|
/// The overall difficulty of the associated beatmap.
|
|
/// </summary>
|
|
float OverallDifficulty { get; }
|
|
|
|
/// <summary>
|
|
/// The approach rate of the associated beatmap.
|
|
/// </summary>
|
|
float ApproachRate { get; }
|
|
|
|
/// <summary>
|
|
/// The base slider velocity of the associated beatmap.
|
|
/// This was known as "SliderMultiplier" in the .osu format and stable editor.
|
|
/// </summary>
|
|
double SliderMultiplier { get; }
|
|
|
|
/// <summary>
|
|
/// The slider tick rate of the associated beatmap.
|
|
/// </summary>
|
|
double SliderTickRate { get; }
|
|
|
|
static float CalculateScaleFromCircleSize(float circleSize)
|
|
{
|
|
// The following comment is copied verbatim from osu-stable:
|
|
//
|
|
// Builds of osu! up to 2013-05-04 had the gamefield being rounded down, which caused incorrect radius calculations
|
|
// in widescreen cases. This ratio adjusts to allow for old replays to work post-fix, which in turn increases the lenience
|
|
// for all plays, but by an amount so small it should only be effective in replays.
|
|
//
|
|
// To match expectations of gameplay we need to apply this multiplier to circle scale. It's weird but is what it is.
|
|
// It works out to under 1 game pixel and is generally not meaningful to gameplay, but is to replay playback accuracy.
|
|
const float broken_gamefield_rounding_allowance = 1.00041f;
|
|
|
|
return (float)(1.0f - 0.7f * DifficultyRange(circleSize)) / 2 * broken_gamefield_rounding_allowance;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Maps a difficulty value [0, 10] to a two-piece linear range of values.
|
|
/// </summary>
|
|
/// <param name="difficulty">The difficulty value to be mapped.</param>
|
|
/// <param name="min">Minimum of the resulting range which will be achieved by a difficulty value of 0.</param>
|
|
/// <param name="mid">Midpoint of the resulting range which will be achieved by a difficulty value of 5.</param>
|
|
/// <param name="max">Maximum of the resulting range which will be achieved by a difficulty value of 10.</param>
|
|
/// <returns>Value to which the difficulty value maps in the specified range.</returns>
|
|
static double DifficultyRange(double difficulty, double min, double mid, double max)
|
|
{
|
|
if (difficulty > 5)
|
|
return mid + (max - mid) * DifficultyRange(difficulty);
|
|
if (difficulty < 5)
|
|
return mid - (mid - min) * (5 - difficulty) / 5;
|
|
|
|
return mid;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Maps a difficulty value [0, 10] to a linear range of [-1, 1].
|
|
/// </summary>
|
|
/// <param name="difficulty">The difficulty value to be mapped.</param>
|
|
/// <returns>Value to which the difficulty value maps in the specified range.</returns>
|
|
static double DifficultyRange(double difficulty) => (difficulty - 5) / 5;
|
|
|
|
/// <summary>
|
|
/// Maps a difficulty value [0, 10] to a two-piece linear range of values.
|
|
/// </summary>
|
|
/// <param name="difficulty">The difficulty value to be mapped.</param>
|
|
/// <param name="range">The values that define the two linear ranges.
|
|
/// <list type="table">
|
|
/// <item>
|
|
/// <term>od0</term>
|
|
/// <description>Minimum of the resulting range which will be achieved by a difficulty value of 0.</description>
|
|
/// </item>
|
|
/// <item>
|
|
/// <term>od5</term>
|
|
/// <description>Midpoint of the resulting range which will be achieved by a difficulty value of 5.</description>
|
|
/// </item>
|
|
/// <item>
|
|
/// <term>od10</term>
|
|
/// <description>Maximum of the resulting range which will be achieved by a difficulty value of 10.</description>
|
|
/// </item>
|
|
/// </list>
|
|
/// </param>
|
|
/// <returns>Value to which the difficulty value maps in the specified range.</returns>
|
|
static double DifficultyRange(double difficulty, (double od0, double od5, double od10) range)
|
|
=> DifficultyRange(difficulty, range.od0, range.od5, range.od10);
|
|
}
|
|
}
|