1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-22 08:32:54 +08:00
osu-lazer/osu.Game.Rulesets.Catch/Beatmaps/CatchBeatmapProcessor.cs

236 lines
8.7 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.Collections.Generic;
using System.Linq;
using osu.Game.Beatmaps;
using osu.Game.Rulesets.Catch.Objects;
using osu.Game.Rulesets.Catch.UI;
using osu.Game.Rulesets.Objects.Types;
2018-11-20 15:51:59 +08:00
using osuTK;
using osu.Game.Rulesets.Catch.MathUtils;
2019-07-31 17:44:01 +08:00
using osu.Game.Rulesets.Mods;
2018-04-13 17:19:50 +08:00
namespace osu.Game.Rulesets.Catch.Beatmaps
{
public class CatchBeatmapProcessor : BeatmapProcessor
2018-04-13 17:19:50 +08:00
{
public const int RNG_SEED = 1337;
public CatchBeatmapProcessor(IBeatmap beatmap)
: base(beatmap)
2018-04-13 17:19:50 +08:00
{
}
public override void PostProcess()
{
base.PostProcess();
2018-04-13 17:19:50 +08:00
2019-07-31 17:44:01 +08:00
ApplyPositionOffsets(Beatmap);
2018-06-29 12:52:13 +08:00
initialiseHyperDash((List<CatchHitObject>)Beatmap.HitObjects);
2018-04-13 17:19:50 +08:00
int index = 0;
2019-04-01 11:16:05 +08:00
foreach (var obj in Beatmap.HitObjects.OfType<CatchHitObject>())
{
2018-04-13 17:19:50 +08:00
obj.IndexInBeatmap = index++;
if (obj.LastInCombo && obj.NestedHitObjects.LastOrDefault() is IHasComboInformation lastNested)
lastNested.LastInCombo = true;
}
2018-04-13 17:19:50 +08:00
}
2019-07-31 17:44:01 +08:00
public static void ApplyPositionOffsets(IBeatmap beatmap, params Mod[] mods)
{
var rng = new FastRandom(RNG_SEED);
2019-07-31 17:44:01 +08:00
bool shouldApplyHardRockOffset = mods.Any(m => m is ModHardRock);
float? lastPosition = null;
double lastStartTime = 0;
foreach (var obj in beatmap.HitObjects.OfType<CatchHitObject>())
{
obj.XOffset = 0;
switch (obj)
{
2019-07-31 17:44:01 +08:00
case Fruit fruit:
if (shouldApplyHardRockOffset)
applyHardRockOffset(fruit, ref lastPosition, ref lastStartTime, rng);
break;
case BananaShower bananaShower:
2018-06-29 14:01:33 +08:00
foreach (var banana in bananaShower.NestedHitObjects.OfType<Banana>())
{
banana.XOffset = (float)rng.NextDouble();
rng.Next(); // osu!stable retrieved a random banana type
rng.Next(); // osu!stable retrieved a random banana rotation
2018-06-13 20:10:54 +08:00
rng.Next(); // osu!stable retrieved a random banana colour
}
2019-02-28 12:31:40 +08:00
break;
2019-04-01 11:16:05 +08:00
case JuiceStream juiceStream:
foreach (var nested in juiceStream.NestedHitObjects)
{
2019-08-01 13:57:17 +08:00
var catchObject = (CatchHitObject)nested;
catchObject.XOffset = 0;
if (catchObject is TinyDroplet)
catchObject.XOffset = MathHelper.Clamp(rng.Next(-20, 20) / CatchPlayfield.BASE_WIDTH, -catchObject.X, 1 - catchObject.X);
else if (catchObject is Droplet)
2018-06-13 18:52:04 +08:00
rng.Next(); // osu!stable retrieved a random droplet rotation
}
2019-02-28 12:31:40 +08:00
break;
}
}
}
private static void applyHardRockOffset(CatchHitObject hitObject, ref float? lastPosition, ref double lastStartTime, FastRandom rng)
2019-07-31 17:44:01 +08:00
{
if (hitObject is JuiceStream stream)
{
lastPosition = stream.EndX;
lastStartTime = stream.EndTime;
return;
}
if (!(hitObject is Fruit))
return;
float offsetPosition = hitObject.X;
2019-07-31 17:44:01 +08:00
double startTime = hitObject.StartTime;
if (lastPosition == null)
{
lastPosition = offsetPosition;
2019-07-31 17:44:01 +08:00
lastStartTime = startTime;
return;
}
float positionDiff = offsetPosition - lastPosition.Value;
2019-07-31 17:44:01 +08:00
double timeDiff = startTime - lastStartTime;
if (timeDiff > 1000)
{
lastPosition = offsetPosition;
2019-07-31 17:44:01 +08:00
lastStartTime = startTime;
return;
}
if (positionDiff == 0)
{
applyRandomOffset(ref offsetPosition, timeDiff / 4d, rng);
2019-08-01 13:57:07 +08:00
hitObject.XOffset = offsetPosition - hitObject.X;
2019-07-31 17:44:01 +08:00
return;
}
if (Math.Abs(positionDiff * CatchPlayfield.BASE_WIDTH) < timeDiff / 3d)
applyOffset(ref offsetPosition, positionDiff);
2019-07-31 17:44:01 +08:00
2019-08-01 13:57:07 +08:00
hitObject.XOffset = offsetPosition - hitObject.X;
2019-07-31 17:44:01 +08:00
lastPosition = offsetPosition;
2019-07-31 17:44:01 +08:00
lastStartTime = startTime;
}
/// <summary>
/// Applies a random offset in a random direction to a position, ensuring that the final position remains within the boundary of the playfield.
/// </summary>
/// <param name="position">The position which the offset should be applied to.</param>
/// <param name="maxOffset">The maximum offset, cannot exceed 20px.</param>
/// <param name="rng">The random number generator.</param>
private static void applyRandomOffset(ref float position, double maxOffset, FastRandom rng)
{
bool right = rng.NextBool();
float rand = Math.Min(20, (float)rng.Next(0, Math.Max(0, maxOffset))) / CatchPlayfield.BASE_WIDTH;
if (right)
{
// Clamp to the right bound
if (position + rand <= 1)
position += rand;
else
position -= rand;
}
else
{
// Clamp to the left bound
if (position - rand >= 0)
position -= rand;
else
position += rand;
}
}
/// <summary>
/// Applies an offset to a position, ensuring that the final position remains within the boundary of the playfield.
/// </summary>
/// <param name="position">The position which the offset should be applied to.</param>
/// <param name="amount">The amount to offset by.</param>
private static void applyOffset(ref float position, float amount)
{
if (amount > 0)
{
// Clamp to the right bound
if (position + amount < 1)
position += amount;
}
else
{
// Clamp to the left bound
if (position + amount > 0)
position += amount;
}
}
2018-04-13 17:19:50 +08:00
private void initialiseHyperDash(List<CatchHitObject> objects)
{
List<CatchHitObject> objectWithDroplets = new List<CatchHitObject>();
2018-04-13 17:19:50 +08:00
foreach (var currentObject in objects)
2018-04-13 17:19:50 +08:00
{
if (currentObject is Fruit)
objectWithDroplets.Add(currentObject);
if (currentObject is JuiceStream)
foreach (var currentJuiceElement in currentObject.NestedHitObjects)
if (!(currentJuiceElement is TinyDroplet))
objectWithDroplets.Add((CatchHitObject)currentJuiceElement);
}
2018-04-13 17:19:50 +08:00
objectWithDroplets.Sort((h1, h2) => h1.StartTime.CompareTo(h2.StartTime));
double halfCatcherWidth = CatcherArea.GetCatcherSize(Beatmap.BeatmapInfo.BaseDifficulty) / 2;
int lastDirection = 0;
double lastExcess = halfCatcherWidth;
2018-04-13 17:19:50 +08:00
for (int i = 0; i < objectWithDroplets.Count - 1; i++)
{
CatchHitObject currentObject = objectWithDroplets[i];
CatchHitObject nextObject = objectWithDroplets[i + 1];
2018-04-13 17:19:50 +08:00
int thisDirection = nextObject.X > currentObject.X ? 1 : -1;
double timeToNext = nextObject.StartTime - currentObject.StartTime - 1000f / 60f / 4; // 1/4th of a frame of grace time, taken from osu-stable
2018-04-13 17:19:50 +08:00
double distanceToNext = Math.Abs(nextObject.X - currentObject.X) - (lastDirection == thisDirection ? lastExcess : halfCatcherWidth);
float distanceToHyper = (float)(timeToNext * CatcherArea.Catcher.BASE_SPEED - distanceToNext);
2019-04-01 11:16:05 +08:00
if (distanceToHyper < 0)
2018-04-13 17:19:50 +08:00
{
currentObject.HyperDashTarget = nextObject;
lastExcess = halfCatcherWidth;
}
else
{
currentObject.DistanceToHyperDash = distanceToHyper;
lastExcess = MathHelper.Clamp(distanceToHyper, 0, halfCatcherWidth);
2018-04-13 17:19:50 +08:00
}
lastDirection = thisDirection;
}
}
}
}