1
0
mirror of https://github.com/ppy/osu.git synced 2024-11-18 02:22:59 +08:00
osu-lazer/osu.Game.Rulesets.Osu/Difficulty/Preprocessing/OsuDifficultyHitObject.cs

236 lines
11 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 17:19:50 +08:00
using System;
using System.Linq;
using osu.Game.Rulesets.Difficulty.Preprocessing;
using osu.Game.Rulesets.Objects;
2018-04-13 17:19:50 +08:00
using osu.Game.Rulesets.Osu.Objects;
2018-11-20 15:51:59 +08:00
using osuTK;
2018-04-13 17:19:50 +08:00
2018-05-15 16:36:29 +08:00
namespace osu.Game.Rulesets.Osu.Difficulty.Preprocessing
2018-04-13 17:19:50 +08:00
{
public class OsuDifficultyHitObject : DifficultyHitObject
2018-04-13 17:19:50 +08:00
{
2021-09-25 11:02:33 +08:00
private const int normalized_radius = 50; // Change radius to 50 to make 100 the diameter. Easier for mental maths.
private const int min_delta_time = 25;
2018-05-15 20:44:45 +08:00
protected new OsuHitObject BaseObject => (OsuHitObject)base.BaseObject;
2018-04-13 17:19:50 +08:00
2021-09-15 18:24:48 +08:00
/// <summary>
2021-09-25 11:02:33 +08:00
/// Normalized distance from the end position of the previous <see cref="OsuDifficultyHitObject"/> to the start position of this <see cref="OsuDifficultyHitObject"/>.
2021-09-15 18:24:48 +08:00
/// </summary>
2021-09-25 11:02:33 +08:00
public double JumpDistance { get; private set; }
2021-09-15 18:24:48 +08:00
2018-04-13 17:19:50 +08:00
/// <summary>
2021-10-13 23:41:24 +08:00
/// Minimum distance from the end position of the previous <see cref="OsuDifficultyHitObject"/> to the start position of this <see cref="OsuDifficultyHitObject"/>.
2018-04-13 17:19:50 +08:00
/// </summary>
2021-10-13 23:41:24 +08:00
public double MovementDistance { get; private set; }
/// <summary>
2018-10-09 11:03:47 +08:00
/// Normalized distance between the start and end position of the previous <see cref="OsuDifficultyHitObject"/>.
/// </summary>
public double TravelDistance { get; private set; }
2018-04-13 17:19:50 +08:00
/// <summary>
/// Angle the player has to take to hit this <see cref="OsuDifficultyHitObject"/>.
/// Calculated as the angle between the circles (current-2, current-1, current).
2018-04-13 17:19:50 +08:00
/// </summary>
public double? Angle { get; private set; }
2018-04-13 17:19:50 +08:00
2021-10-13 23:41:24 +08:00
/// <summary>
2021-11-02 22:47:20 +08:00
/// Milliseconds elapsed since the end time of the previous <see cref="OsuDifficultyHitObject"/>, with a minimum of 25ms.
2021-10-13 23:41:24 +08:00
/// </summary>
public double MovementTime { get; private set; }
/// <summary>
2021-11-02 22:47:20 +08:00
/// Milliseconds elapsed since the start time of the previous <see cref="OsuDifficultyHitObject"/> to the end time of the same previous <see cref="OsuDifficultyHitObject"/>, with a minimum of 25ms.
2021-10-13 23:41:24 +08:00
/// </summary>
public double TravelTime { get; private set; }
2021-09-25 11:02:33 +08:00
/// <summary>
/// Milliseconds elapsed since the start time of the previous <see cref="OsuDifficultyHitObject"/>, with a minimum of 25ms.
2021-09-25 11:02:33 +08:00
/// </summary>
public readonly double StrainTime;
private readonly OsuHitObject lastLastObject;
private readonly OsuHitObject lastObject;
2018-04-13 17:19:50 +08:00
2019-02-19 16:43:12 +08:00
public OsuDifficultyHitObject(HitObject hitObject, HitObject lastLastObject, HitObject lastObject, double clockRate)
: base(hitObject, lastObject, clockRate)
2018-04-13 17:19:50 +08:00
{
this.lastLastObject = (OsuHitObject)lastLastObject;
this.lastObject = (OsuHitObject)lastObject;
2021-11-02 22:47:20 +08:00
// Capped to 25ms to prevent difficulty calculation breaking from simultaneous objects.
StrainTime = Math.Max(DeltaTime, min_delta_time);
2021-10-13 23:41:24 +08:00
setDistances(clockRate);
2018-04-13 17:19:50 +08:00
}
2021-10-13 23:41:24 +08:00
private void setDistances(double clockRate)
2018-04-13 17:19:50 +08:00
{
// We don't need to calculate either angle or distance when one of the last->curr objects is a spinner
if (BaseObject is Spinner || lastObject is Spinner)
return;
2018-04-13 17:19:50 +08:00
// We will scale distances by this factor, so we can assume a uniform CircleSize among beatmaps.
float scalingFactor = normalized_radius / (float)BaseObject.Radius;
2019-04-01 11:16:05 +08:00
2018-04-13 17:19:50 +08:00
if (BaseObject.Radius < 30)
{
2018-12-21 21:52:27 +08:00
float smallCircleBonus = Math.Min(30 - (float)BaseObject.Radius, 5) / 50;
2018-04-13 17:19:50 +08:00
scalingFactor *= 1 + smallCircleBonus;
}
Vector2 lastCursorPosition = getEndCursorPosition(lastObject);
JumpDistance = (BaseObject.StackedPosition * scalingFactor - lastCursorPosition * scalingFactor).Length;
if (lastObject is Slider lastSlider)
{
computeSliderCursorPosition(lastSlider);
2021-10-22 00:08:35 +08:00
TravelDistance = 0;
TravelTime = Math.Max(lastSlider.LazyTravelTime / clockRate, min_delta_time);
MovementTime = Math.Max(StrainTime - TravelTime, min_delta_time);
2021-10-14 00:04:39 +08:00
MovementDistance = Vector2.Subtract(lastSlider.TailCircle.StackedPosition, BaseObject.StackedPosition).Length * scalingFactor;
2021-10-22 00:08:35 +08:00
int repeatCount = 0;
Vector2 currSliderPosition = ((OsuHitObject)lastSlider.NestedHitObjects[0]).StackedPosition;
2021-10-22 00:08:35 +08:00
for (int i = 1; i < lastSlider.NestedHitObjects.Count; i++)
{
Vector2 currSlider = Vector2.Subtract(((OsuHitObject)lastSlider.NestedHitObjects[i]).StackedPosition, currSliderPosition);
double currSliderLength = currSlider.Length * scalingFactor;
2021-10-22 00:08:35 +08:00
if ((OsuHitObject)lastSlider.NestedHitObjects[i] is SliderEndCircle && !((OsuHitObject)lastSlider.NestedHitObjects[i] is SliderRepeat))
{
Vector2 possSlider = Vector2.Subtract((Vector2)lastSlider.LazyEndPosition, currSliderPosition);
if (possSlider.Length < currSlider.Length)
currSlider = possSlider; // Take the least distance from slider end vs lazy end.
currSliderLength = currSlider.Length * scalingFactor;
}
if ((OsuHitObject)lastSlider.NestedHitObjects[i] is SliderTick)
{
if (currSliderLength > 120)
{
currSliderPosition = Vector2.Add(currSliderPosition, Vector2.Multiply(currSlider, (float)((currSliderLength - 120) / currSliderLength)));
currSliderLength *= (currSliderLength - 120) / currSliderLength;
}
else
currSliderLength = 0;
}
else if ((OsuHitObject)lastSlider.NestedHitObjects[i] is SliderRepeat)
2021-10-23 01:17:19 +08:00
{
if (currSliderLength > 50)
{
currSliderPosition = Vector2.Add(currSliderPosition, Vector2.Multiply(currSlider, (float)((currSliderLength - 50) / currSliderLength)));
currSliderLength *= (currSliderLength - 50) / currSliderLength;
}
else
currSliderLength = 0;
2021-10-23 01:17:19 +08:00
}
2021-10-22 00:08:35 +08:00
else
{
if (currSliderLength > 0)
{
currSliderPosition = Vector2.Add(currSliderPosition, Vector2.Multiply(currSlider, (float)((currSliderLength - 0) / currSliderLength)));
currSliderLength *= (currSliderLength - 0) / currSliderLength;
}
else
currSliderLength = 0;
}
2021-10-22 00:08:35 +08:00
if ((OsuHitObject)lastSlider.NestedHitObjects[i] is SliderRepeat)
repeatCount++;
TravelDistance += currSliderLength;
2021-10-22 00:08:35 +08:00
}
TravelDistance *= Math.Pow(1 + repeatCount / 2.5, 1.0 / 2.5); // Bonus for repeat sliders until a better per nested object strain system can be achieved.
// Jump distance from the slider tail to the next object, as opposed to the lazy position of JumpDistance.
float tailJumpDistance = Vector2.Subtract(lastSlider.TailCircle.StackedPosition, BaseObject.StackedPosition).Length * scalingFactor;
2018-04-13 17:19:50 +08:00
// For hitobjects which continue in the direction of the slider, the player will normally follow through the slider,
// such that they're not jumping from the lazy position but rather from very close to (or the end of) the slider.
// In such cases, a leniency is applied by also considering the jump distance from the tail of the slider, and taking the minimum jump distance.
// Additional distance is removed based on position of jump relative to slider follow circle radius.
// JumpDistance is 50 since follow radius = 1.4 * radius. tailJumpDistance is 120 since the full distance of radial leniency is still possible.
MovementDistance = Math.Max(0, Math.Min(JumpDistance - 50, tailJumpDistance - 120));
}
else
{
MovementTime = StrainTime;
MovementDistance = JumpDistance;
}
2018-12-08 14:01:26 +08:00
if (lastLastObject != null && !(lastLastObject is Spinner))
2018-12-08 14:01:26 +08:00
{
Vector2 lastLastCursorPosition = getEndCursorPosition(lastLastObject);
2018-12-08 14:01:26 +08:00
Vector2 v1 = lastLastCursorPosition - lastObject.StackedPosition;
Vector2 v2 = BaseObject.StackedPosition - lastCursorPosition;
2018-12-08 14:01:26 +08:00
float dot = Vector2.Dot(v1, v2);
float det = v1.X * v2.Y - v1.Y * v2.X;
2018-12-09 19:31:04 +08:00
Angle = Math.Abs(Math.Atan2(det, dot));
2018-12-08 14:01:26 +08:00
}
2018-04-13 17:19:50 +08:00
}
private void computeSliderCursorPosition(Slider slider)
{
if (slider.LazyEndPosition != null)
return;
2019-02-28 12:31:40 +08:00
2018-10-11 12:53:29 +08:00
slider.LazyEndPosition = slider.StackedPosition;
2018-04-13 17:19:50 +08:00
2021-10-23 01:17:19 +08:00
float approxFollowCircleRadius = (float)(slider.Radius * 1.4); // using 1.4 to better follow the real movement of a cursor.
2018-04-13 17:19:50 +08:00
var computeVertex = new Action<double>(t =>
{
double progress = (t - slider.StartTime) / slider.SpanDuration;
if (progress % 2 >= 1)
2018-10-08 17:37:30 +08:00
progress = 1 - progress % 1;
else
2019-11-12 17:56:38 +08:00
progress %= 1;
2018-10-08 17:37:30 +08:00
2018-04-13 17:19:50 +08:00
// ReSharper disable once PossibleInvalidOperationException (bugged in current r# version)
var diff = slider.StackedPosition + slider.Path.PositionAt(progress) - slider.LazyEndPosition.Value;
2018-04-13 17:19:50 +08:00
float dist = diff.Length;
2021-10-13 23:41:24 +08:00
slider.LazyTravelTime = t - slider.StartTime;
2021-10-23 01:17:19 +08:00
if (dist > approxFollowCircleRadius)
2018-04-13 17:19:50 +08:00
{
// The cursor would be outside the follow circle, we need to move it
diff.Normalize(); // Obtain direction of diff
2021-10-23 01:17:19 +08:00
dist -= approxFollowCircleRadius;
2018-04-13 17:19:50 +08:00
slider.LazyEndPosition += diff * dist;
slider.LazyTravelDistance += dist;
}
});
// Skip the head circle
var scoringTimes = slider.NestedHitObjects.Skip(1).Select(t => t.StartTime);
foreach (double time in scoringTimes)
2018-04-13 17:19:50 +08:00
computeVertex(time);
}
private Vector2 getEndCursorPosition(OsuHitObject hitObject)
{
Vector2 pos = hitObject.StackedPosition;
2019-02-28 13:35:00 +08:00
if (hitObject is Slider slider)
{
computeSliderCursorPosition(slider);
pos = slider.LazyEndPosition ?? pos;
}
return pos;
}
2018-04-13 17:19:50 +08:00
}
}