osu/osu.Game.Rulesets.Osu/Difficulty/Skills/Aim.cs

79 lines
2.9 KiB
C#
Raw Normal View History

// 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.
2018-04-13 09:19:50 +00:00
using System;
using osu.Game.Rulesets.Difficulty.Preprocessing;
using osu.Game.Rulesets.Mods;
2018-05-15 08:36:29 +00:00
using osu.Game.Rulesets.Osu.Difficulty.Preprocessing;
2019-02-18 05:58:33 +00:00
using osu.Game.Rulesets.Osu.Objects;
2018-04-13 09:19:50 +00:00
2018-05-15 08:36:29 +00:00
namespace osu.Game.Rulesets.Osu.Difficulty.Skills
2018-04-13 09:19:50 +00:00
{
/// <summary>
/// Represents the skill required to correctly aim at every object in the map with a uniform CircleSize and normalized distances.
/// </summary>
2021-06-14 17:18:49 +00:00
public class Aim : OsuStrainSkill
2018-04-13 09:19:50 +00:00
{
2018-12-24 03:41:04 +00:00
private const double angle_bonus_begin = Math.PI / 3;
2018-12-22 00:31:30 +00:00
private const double timing_threshold = 107;
2018-12-22 00:56:33 +00:00
public Aim(Mod[] mods)
: base(mods)
{
}
2021-08-17 13:39:18 +00:00
private double currentStrain = 1;
2018-04-13 09:19:50 +00:00
2021-08-17 13:39:18 +00:00
private double skillMultiplier => 26.25;
private double strainDecayBase => 0.15;
private double strainValueOf(DifficultyHitObject current)
2018-12-08 06:01:26 +00:00
{
2019-02-18 05:58:33 +00:00
if (current.BaseObject is Spinner)
return 0;
var osuCurrent = (OsuDifficultyHitObject)current;
2018-12-24 03:41:04 +00:00
2021-08-17 13:39:18 +00:00
double aimStrain = 0;
2019-01-29 07:35:20 +00:00
2018-12-21 05:52:43 +00:00
if (Previous.Count > 0)
{
var osuPrevious = (OsuDifficultyHitObject)Previous[0];
if (osuCurrent.Angle != null && osuCurrent.Angle.Value > angle_bonus_begin)
2018-12-21 13:52:27 +00:00
{
const double scale = 90;
2018-12-24 03:41:04 +00:00
var angleBonus = Math.Sqrt(
Math.Max(osuPrevious.JumpDistance - scale, 0)
* Math.Pow(Math.Sin(osuCurrent.Angle.Value - angle_bonus_begin), 2)
* Math.Max(osuCurrent.JumpDistance - scale, 0));
2021-08-17 13:39:18 +00:00
aimStrain = 1.4 * applyDiminishingExp(Math.Max(0, angleBonus)) / Math.Max(timing_threshold, osuPrevious.StrainTime);
2018-12-21 13:52:27 +00:00
}
2018-12-21 05:52:43 +00:00
}
double jumpDistanceExp = applyDiminishingExp(osuCurrent.JumpDistance);
double travelDistanceExp = applyDiminishingExp(osuCurrent.TravelDistance);
2019-01-29 07:35:20 +00:00
2018-12-24 03:41:04 +00:00
return Math.Max(
2021-08-17 13:39:18 +00:00
aimStrain + (jumpDistanceExp + travelDistanceExp + Math.Sqrt(travelDistanceExp * jumpDistanceExp)) / Math.Max(osuCurrent.StrainTime, timing_threshold),
(Math.Sqrt(travelDistanceExp * jumpDistanceExp) + jumpDistanceExp + travelDistanceExp) / osuCurrent.StrainTime
2019-01-29 07:35:20 +00:00
);
2018-12-08 06:01:26 +00:00
}
private double applyDiminishingExp(double val) => Math.Pow(val, 0.99);
2021-08-17 13:39:18 +00:00
private double strainDecay(double ms) => Math.Pow(strainDecayBase, ms / 1000);
protected override double CalculateInitialStrain(double time) => currentStrain * strainDecay(time - Previous[0].StartTime);
protected override double StrainValueAt(DifficultyHitObject current)
{
currentStrain *= strainDecay(current.DeltaTime);
currentStrain += strainValueOf(current) * skillMultiplier;
2021-08-17 13:39:18 +00:00
return currentStrain;
}
2018-04-13 09:19:50 +00:00
}
}