mirror of
https://github.com/ppy/osu.git
synced 2025-01-26 18:52:55 +08:00
changed decay system to allow for customizing the currentStrain
This commit is contained in:
parent
7d46b3f9c5
commit
176b3e7533
@ -9,7 +9,7 @@ using osu.Game.Rulesets.Mods;
|
|||||||
|
|
||||||
namespace osu.Game.Rulesets.Catch.Difficulty.Skills
|
namespace osu.Game.Rulesets.Catch.Difficulty.Skills
|
||||||
{
|
{
|
||||||
public class Movement : StrainSkill
|
public class Movement : StrainDecaySkill
|
||||||
{
|
{
|
||||||
private const float absolute_player_positioning_error = 16f;
|
private const float absolute_player_positioning_error = 16f;
|
||||||
private const float normalized_hitobject_radius = 41.0f;
|
private const float normalized_hitobject_radius = 41.0f;
|
||||||
|
@ -10,7 +10,7 @@ using osu.Game.Rulesets.Mods;
|
|||||||
|
|
||||||
namespace osu.Game.Rulesets.Mania.Difficulty.Skills
|
namespace osu.Game.Rulesets.Mania.Difficulty.Skills
|
||||||
{
|
{
|
||||||
public class Strain : StrainSkill
|
public class Strain : StrainDecaySkill
|
||||||
{
|
{
|
||||||
private const double individual_decay_base = 0.125;
|
private const double individual_decay_base = 0.125;
|
||||||
private const double overall_decay_base = 0.30;
|
private const double overall_decay_base = 0.30;
|
||||||
@ -71,7 +71,7 @@ namespace osu.Game.Rulesets.Mania.Difficulty.Skills
|
|||||||
return individualStrain + overallStrain - CurrentStrain;
|
return individualStrain + overallStrain - CurrentStrain;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override double GetPeakStrain(double offset)
|
protected override double CalculateInitialStrain(double offset)
|
||||||
=> applyDecay(individualStrain, offset - Previous[0].StartTime, individual_decay_base)
|
=> applyDecay(individualStrain, offset - Previous[0].StartTime, individual_decay_base)
|
||||||
+ applyDecay(overallStrain, offset - Previous[0].StartTime, overall_decay_base);
|
+ applyDecay(overallStrain, offset - Previous[0].StartTime, overall_decay_base);
|
||||||
|
|
||||||
|
@ -12,7 +12,7 @@ namespace osu.Game.Rulesets.Osu.Difficulty.Skills
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Represents the skill required to correctly aim at every object in the map with a uniform CircleSize and normalized distances.
|
/// Represents the skill required to correctly aim at every object in the map with a uniform CircleSize and normalized distances.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class Aim : OsuStrainSkill
|
public class Aim : OsuStrainDecaySkill
|
||||||
{
|
{
|
||||||
private const double angle_bonus_begin = Math.PI / 3;
|
private const double angle_bonus_begin = Math.PI / 3;
|
||||||
private const double timing_threshold = 107;
|
private const double timing_threshold = 107;
|
||||||
|
@ -10,7 +10,7 @@ using osu.Framework.Utils;
|
|||||||
|
|
||||||
namespace osu.Game.Rulesets.Osu.Difficulty.Skills
|
namespace osu.Game.Rulesets.Osu.Difficulty.Skills
|
||||||
{
|
{
|
||||||
public abstract class OsuStrainSkill : StrainSkill
|
public abstract class OsuStrainDecaySkill : StrainDecaySkill
|
||||||
{
|
{
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// The number of sections with the highest strains, which the peak strain reductions will apply to.
|
/// The number of sections with the highest strains, which the peak strain reductions will apply to.
|
||||||
@ -28,7 +28,7 @@ namespace osu.Game.Rulesets.Osu.Difficulty.Skills
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
protected virtual double DifficultyMultiplier => 1.06;
|
protected virtual double DifficultyMultiplier => 1.06;
|
||||||
|
|
||||||
protected OsuStrainSkill(Mod[] mods)
|
protected OsuStrainDecaySkill(Mod[] mods)
|
||||||
: base(mods)
|
: base(mods)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
@ -12,7 +12,7 @@ namespace osu.Game.Rulesets.Osu.Difficulty.Skills
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Represents the skill required to press keys with regards to keeping up with the speed at which objects need to be hit.
|
/// Represents the skill required to press keys with regards to keeping up with the speed at which objects need to be hit.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class Speed : OsuStrainSkill
|
public class Speed : OsuStrainDecaySkill
|
||||||
{
|
{
|
||||||
private const double single_spacing_threshold = 125;
|
private const double single_spacing_threshold = 125;
|
||||||
|
|
||||||
@ -151,14 +151,5 @@ namespace osu.Game.Rulesets.Osu.Difficulty.Skills
|
|||||||
|
|
||||||
return (1 + (speedBonus - 1) * 0.75) * angleBonus * (0.95 + speedBonus * Math.Pow(distance / single_spacing_threshold, 3.5)) / osuCurrent.StrainTime;
|
return (1 + (speedBonus - 1) * 0.75) * angleBonus * (0.95 + speedBonus * Math.Pow(distance / single_spacing_threshold, 3.5)) / osuCurrent.StrainTime;
|
||||||
}
|
}
|
||||||
protected override double GetTotalCurrentStrain(DifficultyHitObject current)
|
|
||||||
{
|
|
||||||
return base.GetTotalCurrentStrain(current) * calculateRhythmBonus(current.StartTime);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected override double GetPeakStrain(double time)
|
|
||||||
{
|
|
||||||
return base.GetPeakStrain(time);// * calculateRhythmBonus(current.StartTime);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -14,7 +14,7 @@ namespace osu.Game.Rulesets.Taiko.Difficulty.Skills
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Calculates the colour coefficient of taiko difficulty.
|
/// Calculates the colour coefficient of taiko difficulty.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class Colour : StrainSkill
|
public class Colour : StrainDecaySkill
|
||||||
{
|
{
|
||||||
protected override double SkillMultiplier => 1;
|
protected override double SkillMultiplier => 1;
|
||||||
protected override double StrainDecayBase => 0.4;
|
protected override double StrainDecayBase => 0.4;
|
||||||
|
@ -14,7 +14,7 @@ namespace osu.Game.Rulesets.Taiko.Difficulty.Skills
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Calculates the rhythm coefficient of taiko difficulty.
|
/// Calculates the rhythm coefficient of taiko difficulty.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class Rhythm : StrainSkill
|
public class Rhythm : StrainDecaySkill
|
||||||
{
|
{
|
||||||
protected override double SkillMultiplier => 10;
|
protected override double SkillMultiplier => 10;
|
||||||
protected override double StrainDecayBase => 0;
|
protected override double StrainDecayBase => 0;
|
||||||
|
@ -17,7 +17,7 @@ namespace osu.Game.Rulesets.Taiko.Difficulty.Skills
|
|||||||
/// <remarks>
|
/// <remarks>
|
||||||
/// The reference play style chosen uses two hands, with full alternating (the hand changes after every hit).
|
/// The reference play style chosen uses two hands, with full alternating (the hand changes after every hit).
|
||||||
/// </remarks>
|
/// </remarks>
|
||||||
public class Stamina : StrainSkill
|
public class Stamina : StrainDecaySkill
|
||||||
{
|
{
|
||||||
protected override double SkillMultiplier => 1;
|
protected override double SkillMultiplier => 1;
|
||||||
protected override double StrainDecayBase => 0.4;
|
protected override double StrainDecayBase => 0.4;
|
||||||
|
64
osu.Game/Rulesets/Difficulty/Skills/StrainDecaySkill.cs
Normal file
64
osu.Game/Rulesets/Difficulty/Skills/StrainDecaySkill.cs
Normal file
@ -0,0 +1,64 @@
|
|||||||
|
// 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 StrainDecaySkill : StrainSkill
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Strain values are multiplied by this number for the given skill. Used to balance the value of different skills between each other.
|
||||||
|
/// </summary>
|
||||||
|
protected abstract double SkillMultiplier { get; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Determines how quickly strain decays for the given skill.
|
||||||
|
/// For example a value of 0.15 indicates that strain decays to 15% of its original value in one second.
|
||||||
|
/// </summary>
|
||||||
|
protected abstract double StrainDecayBase { get; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// The current strain level.
|
||||||
|
/// </summary>
|
||||||
|
protected double CurrentStrain { get; private set; } = 1;
|
||||||
|
|
||||||
|
protected StrainDecaySkill(Mod[] mods)
|
||||||
|
: base(mods)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Retrieves the peak strain at a point in time.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="time">The time to retrieve the peak strain at.</param>
|
||||||
|
/// <returns>The peak strain.</returns>
|
||||||
|
protected override double CalculateInitialStrain(double time) => CurrentStrain * strainDecay(time - Previous[0].StartTime);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Returns the strain value of <see cref="DifficultyHitObject"/>. This value is calculated with or without respect to previous objects.
|
||||||
|
/// </summary>
|
||||||
|
protected override double StrainValueAt(DifficultyHitObject current)
|
||||||
|
{
|
||||||
|
CurrentStrain *= strainDecay(current.DeltaTime);
|
||||||
|
CurrentStrain += StrainValueOf(current) * SkillMultiplier;
|
||||||
|
|
||||||
|
return CurrentStrain;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Calculates the strain value of a <see cref="DifficultyHitObject"/>. This value is affected by previously processed objects.
|
||||||
|
/// </summary>
|
||||||
|
protected abstract double StrainValueOf(DifficultyHitObject current);
|
||||||
|
|
||||||
|
private double strainDecay(double ms) => Math.Pow(StrainDecayBase, ms / 1000);
|
||||||
|
}
|
||||||
|
}
|
@ -15,27 +15,11 @@ namespace osu.Game.Rulesets.Difficulty.Skills
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public abstract class StrainSkill : Skill
|
public abstract class StrainSkill : Skill
|
||||||
{
|
{
|
||||||
/// <summary>
|
|
||||||
/// Strain values are multiplied by this number for the given skill. Used to balance the value of different skills between each other.
|
|
||||||
/// </summary>
|
|
||||||
protected abstract double SkillMultiplier { get; }
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Determines how quickly strain decays for the given skill.
|
|
||||||
/// For example a value of 0.15 indicates that strain decays to 15% of its original value in one second.
|
|
||||||
/// </summary>
|
|
||||||
protected abstract double StrainDecayBase { get; }
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// The weight by which each strain value decays.
|
/// The weight by which each strain value decays.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
protected virtual double DecayWeight => 0.9;
|
protected virtual double DecayWeight => 0.9;
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// The current strain level.
|
|
||||||
/// </summary>
|
|
||||||
protected double CurrentStrain { get; private set; } = 1;
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// The length of each strain section.
|
/// The length of each strain section.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@ -52,6 +36,11 @@ namespace osu.Game.Rulesets.Difficulty.Skills
|
|||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Returns the strain value of <see cref="DifficultyHitObject"/>. This value is calculated with or without respect to previous objects.
|
||||||
|
/// </summary>
|
||||||
|
protected abstract double StrainValueAt(DifficultyHitObject current);
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Process a <see cref="DifficultyHitObject"/> and update current strain values accordingly.
|
/// Process a <see cref="DifficultyHitObject"/> and update current strain values accordingly.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@ -68,15 +57,7 @@ namespace osu.Game.Rulesets.Difficulty.Skills
|
|||||||
currentSectionEnd += SectionLength;
|
currentSectionEnd += SectionLength;
|
||||||
}
|
}
|
||||||
|
|
||||||
CurrentStrain *= strainDecay(current.DeltaTime);
|
currentSectionPeak = Math.Max(StrainValueAt(current), currentSectionPeak);
|
||||||
CurrentStrain += StrainValueOf(current) * SkillMultiplier;
|
|
||||||
|
|
||||||
currentSectionPeak = Math.Max(GetTotalCurrentStrain(current), currentSectionPeak);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected virtual double GetTotalCurrentStrain(DifficultyHitObject current)
|
|
||||||
{
|
|
||||||
return CurrentStrain;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -93,9 +74,9 @@ namespace osu.Game.Rulesets.Difficulty.Skills
|
|||||||
/// <param name="time">The beginning of the new section in milliseconds.</param>
|
/// <param name="time">The beginning of the new section in milliseconds.</param>
|
||||||
private void startNewSectionFrom(double time)
|
private void startNewSectionFrom(double time)
|
||||||
{
|
{
|
||||||
// The maximum strain of the new section is not zero by default, strain decays as usual regardless of section boundaries.
|
// The maximum strain of the new section is not zero by default
|
||||||
// This means we need to capture the strain level at the beginning of the new section, and use that as the initial peak level.
|
// This means we need to capture the strain level at the beginning of the new section, and use that as the initial peak level.
|
||||||
currentSectionPeak = GetPeakStrain(time);
|
currentSectionPeak = CalculateInitialStrain(time);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -103,7 +84,7 @@ namespace osu.Game.Rulesets.Difficulty.Skills
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="time">The time to retrieve the peak strain at.</param>
|
/// <param name="time">The time to retrieve the peak strain at.</param>
|
||||||
/// <returns>The peak strain.</returns>
|
/// <returns>The peak strain.</returns>
|
||||||
protected virtual double GetPeakStrain(double time) => CurrentStrain * strainDecay(time - Previous[0].StartTime);
|
protected abstract double CalculateInitialStrain(double time);
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Returns a live enumerable of the peak strains for each <see cref="SectionLength"/> section of the beatmap,
|
/// Returns a live enumerable of the peak strains for each <see cref="SectionLength"/> section of the beatmap,
|
||||||
@ -129,12 +110,5 @@ namespace osu.Game.Rulesets.Difficulty.Skills
|
|||||||
|
|
||||||
return difficulty;
|
return difficulty;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Calculates the strain value of a <see cref="DifficultyHitObject"/>. This value is affected by previously processed objects.
|
|
||||||
/// </summary>
|
|
||||||
protected abstract double StrainValueOf(DifficultyHitObject current);
|
|
||||||
|
|
||||||
private double strainDecay(double ms) => Math.Pow(StrainDecayBase, ms / 1000);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user