1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-19 15:02:54 +08:00

Add slider position randomisation

This commit is contained in:
Pasi4K5 2021-05-14 23:04:09 +02:00
parent dbc2318710
commit 3fa6a0413b

View File

@ -53,9 +53,10 @@ namespace osu.Game.Rulesets.Osu.Mods
var prevObjectInfo = new HitObjectInfo var prevObjectInfo = new HitObjectInfo
{ {
AngleRad = 0, StartPosUnchanged = hitObjects[0].Position,
PosUnchanged = hitObjects[0].Position, EndPosUnchanged = hitObjects[0].EndPosition,
PosChanged = hitObjects[0].Position StartPosChanged = hitObjects[0].Position,
EndPosChanged = hitObjects[0].EndPosition
}; };
float rateOfChangeMultiplier = 0; float rateOfChangeMultiplier = 0;
@ -66,31 +67,51 @@ namespace osu.Game.Rulesets.Osu.Mods
var currentObjectInfo = new HitObjectInfo var currentObjectInfo = new HitObjectInfo
{ {
AngleRad = 0, StartPosUnchanged = h.Position,
PosUnchanged = h.EndPosition, EndPosUnchanged = h.EndPosition,
PosChanged = Vector2.Zero StartPosChanged = Vector2.Zero,
EndPosChanged = Vector2.Zero
}; };
// rateOfChangeMultiplier only changes every i iterations to prevent shaky-line-shaped streams // rateOfChangeMultiplier only changes every i iterations to prevent shaky-line-shaped streams
if (i % 3 == 0) if (i % 3 == 0)
rateOfChangeMultiplier = (float)rng.NextDouble() * 2 - 1; rateOfChangeMultiplier = (float)rng.NextDouble() * 2 - 1;
if (h is HitCircle circle) var distanceToPrev = Vector2.Distance(prevObjectInfo.EndPosUnchanged, currentObjectInfo.StartPosUnchanged);
switch (h)
{ {
var distanceToPrev = Vector2.Distance(currentObjectInfo.PosUnchanged, prevObjectInfo.PosUnchanged); case HitCircle circle:
getObjectInfo(
rateOfChangeMultiplier,
prevObjectInfo,
distanceToPrev,
ref currentObjectInfo
);
getObjectInfo( circle.Position = currentObjectInfo.StartPosChanged;
rateOfChangeMultiplier, currentObjectInfo.EndPosChanged = currentObjectInfo.StartPosChanged;
prevObjectInfo, break;
distanceToPrev,
ref currentObjectInfo
);
circle.Position = currentObjectInfo.PosChanged; case Slider slider:
currentObjectInfo.EndPosUnchanged = slider.EndPosition;
currentObjectInfo.EndPosUnchanged = slider.TailCircle.Position;
getObjectInfo(
rateOfChangeMultiplier,
prevObjectInfo,
distanceToPrev,
ref currentObjectInfo
);
slider.Position = currentObjectInfo.StartPosChanged;
currentObjectInfo.EndPosChanged = slider.TailCircle.Position;
moveSliderIntoPlayfield(ref slider, ref currentObjectInfo);
break;
} }
// TODO: Implement slider position randomisation
prevObjectInfo = currentObjectInfo; prevObjectInfo = currentObjectInfo;
} }
} }
@ -116,10 +137,10 @@ namespace osu.Game.Rulesets.Osu.Mods
distanceToPrev * (float)Math.Sin(currentObjectInfo.AngleRad) distanceToPrev * (float)Math.Sin(currentObjectInfo.AngleRad)
); );
posRelativeToPrev = getRotatedVector(prevObjectInfo.PosChanged, posRelativeToPrev); posRelativeToPrev = getRotatedVector(prevObjectInfo.EndPosChanged, posRelativeToPrev);
currentObjectInfo.AngleRad = (float)Math.Atan2(posRelativeToPrev.Y, posRelativeToPrev.X); currentObjectInfo.AngleRad = (float)Math.Atan2(posRelativeToPrev.Y, posRelativeToPrev.X);
var position = Vector2.Add(prevObjectInfo.PosChanged, posRelativeToPrev); var position = Vector2.Add(prevObjectInfo.EndPosChanged, posRelativeToPrev);
// Move hit objects back into the playfield if they are outside of it, // Move hit objects back into the playfield if they are outside of it,
// which would sometimes happen during big jumps otherwise. // which would sometimes happen during big jumps otherwise.
@ -133,7 +154,30 @@ namespace osu.Game.Rulesets.Osu.Mods
else if (position.Y > OsuPlayfield.BASE_SIZE.Y) else if (position.Y > OsuPlayfield.BASE_SIZE.Y)
position.Y = OsuPlayfield.BASE_SIZE.Y; position.Y = OsuPlayfield.BASE_SIZE.Y;
currentObjectInfo.PosChanged = position; currentObjectInfo.StartPosChanged = position;
}
private void moveSliderIntoPlayfield(ref Slider slider, ref HitObjectInfo currentObjectInfo)
{
foreach (var controlPoint in slider.Path.ControlPoints)
{
// Position of controlPoint relative to slider.Position
var pos = controlPoint.Position.Value;
var playfieldSize = OsuPlayfield.BASE_SIZE;
if (pos.X + slider.Position.X < 0)
slider.Position = new Vector2(-pos.X, slider.Position.Y);
else if (pos.X + slider.Position.X > playfieldSize.X)
slider.Position = new Vector2(playfieldSize.X - pos.X, slider.Position.Y);
if (pos.Y + slider.Position.Y < 0)
slider.Position = new Vector2(slider.Position.X, -pos.Y);
else if (pos.Y + slider.Position.Y > playfieldSize.Y)
slider.Position = new Vector2(slider.Position.X, playfieldSize.Y - pos.Y);
}
currentObjectInfo.EndPosChanged = slider.TailCircle.Position;
} }
/// <summary> /// <summary>
@ -217,8 +261,10 @@ namespace osu.Game.Rulesets.Osu.Mods
private struct HitObjectInfo private struct HitObjectInfo
{ {
internal float AngleRad { get; set; } internal float AngleRad { get; set; }
internal Vector2 PosUnchanged { get; set; } internal Vector2 StartPosUnchanged { get; set; }
internal Vector2 PosChanged { get; set; } internal Vector2 EndPosUnchanged { get; set; }
internal Vector2 StartPosChanged { get; set; }
internal Vector2 EndPosChanged { get; set; }
} }
} }