osu/osu.Game.Rulesets.Mania/Beatmaps/ManiaBeatmapConverter.cs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

291 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 09:19:50 +00:00
2017-04-18 07:05:58 +00:00
using osu.Game.Rulesets.Mania.Objects;
2017-04-17 06:44:46 +00:00
using System;
using System.Linq;
using System.Collections.Generic;
using System.Threading;
using osu.Game.Beatmaps;
2017-04-18 07:05:58 +00:00
using osu.Game.Rulesets.Objects;
using osu.Game.Rulesets.Objects.Types;
2017-05-19 07:31:05 +00:00
using osu.Game.Rulesets.Mania.Beatmaps.Patterns;
using osu.Game.Rulesets.Mania.Beatmaps.Patterns.Legacy;
using osu.Game.Rulesets.Mods;
using osu.Game.Rulesets.Scoring.Legacy;
using osu.Game.Utils;
2018-11-20 07:51:59 +00:00
using osuTK;
2018-04-13 09:19:50 +00:00
2017-04-18 07:05:58 +00:00
namespace osu.Game.Rulesets.Mania.Beatmaps
2017-03-11 15:34:21 +00:00
{
public class ManiaBeatmapConverter : BeatmapConverter<ManiaHitObject>
2017-03-11 15:34:21 +00:00
{
/// <summary>
/// Maximum number of previous notes to consider for density calculation.
/// </summary>
private const int max_notes_for_density = 7;
2018-04-13 09:19:50 +00:00
/// <summary>
/// The total number of columns.
/// </summary>
public int TotalColumns => TargetColumns * (Dual ? 2 : 1);
/// <summary>
/// The number of columns per-stage.
/// </summary>
public int TargetColumns;
/// <summary>
/// Whether to double the number of stages.
/// </summary>
2019-02-28 14:40:03 +00:00
public bool Dual;
/// <summary>
/// Whether the beatmap instantiated with is for the mania ruleset.
/// </summary>
public readonly bool IsForCurrentRuleset;
2018-04-13 09:19:50 +00:00
2018-06-15 11:52:09 +00:00
// Internal for testing purposes
internal readonly LegacyRandom Random;
2018-06-15 11:52:09 +00:00
2017-05-19 07:31:05 +00:00
private Pattern lastPattern = new Pattern();
2018-04-13 09:19:50 +00:00
public ManiaBeatmapConverter(IBeatmap beatmap, Ruleset ruleset)
: this(beatmap, LegacyBeatmapConversionDifficultyInfo.FromBeatmap(beatmap), ruleset)
{
}
private ManiaBeatmapConverter(IBeatmap? beatmap, LegacyBeatmapConversionDifficultyInfo difficulty, Ruleset ruleset)
: base(beatmap!, ruleset)
2017-03-11 15:34:21 +00:00
{
IsForCurrentRuleset = difficulty.SourceRuleset.Equals(ruleset.RulesetInfo);
Random = new LegacyRandom((int)MathF.Round(difficulty.DrainRate + difficulty.CircleSize) * 20 + (int)(difficulty.OverallDifficulty * 41.2) + (int)MathF.Round(difficulty.ApproachRate));
TargetColumns = getColumnCount(difficulty);
2018-05-07 02:23:29 +00:00
if (IsForCurrentRuleset && TargetColumns > ManiaRuleset.MAX_STAGE_KEYS)
{
TargetColumns /= 2;
Dual = true;
}
static int getColumnCount(LegacyBeatmapConversionDifficultyInfo difficulty)
{
double roundedCircleSize = Math.Round(difficulty.CircleSize);
if (difficulty.SourceRuleset.ShortName == ManiaRuleset.SHORT_NAME)
return (int)Math.Max(1, roundedCircleSize);
double roundedOverallDifficulty = Math.Round(difficulty.OverallDifficulty);
if (difficulty.TotalObjectCount > 0 && difficulty.EndTimeObjectCount >= 0)
{
int countSliderOrSpinner = difficulty.EndTimeObjectCount;
// In osu!stable, this division appears as if it happens on floats, but due to release-mode
// optimisations, it actually ends up happening on doubles.
double percentSpecialObjects = (double)countSliderOrSpinner / difficulty.TotalObjectCount;
if (percentSpecialObjects < 0.2)
return 7;
if (percentSpecialObjects < 0.3 || roundedCircleSize >= 5)
return roundedOverallDifficulty > 5 ? 7 : 6;
if (percentSpecialObjects > 0.6)
return roundedOverallDifficulty > 4 ? 5 : 4;
}
return Math.Max(4, Math.Min((int)roundedOverallDifficulty + 1, 7));
}
}
public static int GetColumnCount(LegacyBeatmapConversionDifficultyInfo difficulty, IReadOnlyList<Mod>? mods = null)
{
var converter = new ManiaBeatmapConverter(null, difficulty, new ManiaRuleset());
2018-04-13 09:19:50 +00:00
if (mods != null)
{
foreach (var m in mods.OfType<IApplicableToBeatmapConverter>())
m.ApplyToBeatmapConverter(converter);
}
2018-04-13 09:19:50 +00:00
return converter.TotalColumns;
2017-03-11 15:34:21 +00:00
}
2018-04-13 09:19:50 +00:00
public override bool CanConvert() => Beatmap.HitObjects.All(h => h is IHasXPosition);
2019-02-28 10:07:43 +00:00
protected override Beatmap<ManiaHitObject> CreateBeatmap()
{
2024-03-28 13:47:43 +00:00
ManiaBeatmap beatmap = new ManiaBeatmap(new StageDefinition(TargetColumns));
2019-02-28 10:07:43 +00:00
2019-03-01 05:30:58 +00:00
if (Dual)
beatmap.Stages.Add(new StageDefinition(TargetColumns));
2019-02-28 10:07:43 +00:00
return beatmap;
}
2018-04-13 09:19:50 +00:00
protected override IEnumerable<ManiaHitObject> ConvertHitObject(HitObject original, IBeatmap beatmap, CancellationToken cancellationToken)
{
2019-02-28 05:35:00 +00:00
if (original is ManiaHitObject maniaOriginal)
{
yield return maniaOriginal;
2019-02-28 04:31:40 +00:00
yield break;
}
2018-04-13 09:19:50 +00:00
var objects = IsForCurrentRuleset ? generateSpecific(original, beatmap) : generateConverted(original, beatmap);
2017-05-19 07:31:05 +00:00
foreach (ManiaHitObject obj in objects)
yield return obj;
}
2018-04-13 09:19:50 +00:00
private readonly LimitedCapacityQueue<double> prevNoteTimes = new LimitedCapacityQueue<double>(max_notes_for_density);
private double density = int.MaxValue;
2019-02-28 04:31:40 +00:00
private void computeDensity(double newNoteTime)
{
prevNoteTimes.Enqueue(newNoteTime);
2018-04-13 09:19:50 +00:00
2020-10-09 12:11:12 +00:00
if (prevNoteTimes.Count >= 2)
density = (prevNoteTimes[^1] - prevNoteTimes[0]) / prevNoteTimes.Count;
}
2018-04-13 09:19:50 +00:00
private double lastTime;
private Vector2 lastPosition;
2018-06-15 13:08:24 +00:00
private PatternType lastStair = PatternType.Stair;
2019-02-28 04:31:40 +00:00
private void recordNote(double time, Vector2 position)
{
lastTime = time;
lastPosition = position;
}
2018-04-13 09:19:50 +00:00
2017-05-19 07:39:02 +00:00
/// <summary>
/// Method that generates hit objects for osu!mania specific beatmaps.
/// </summary>
/// <param name="original">The original hit object.</param>
/// <param name="originalBeatmap">The original beatmap. This is used to look-up any values dependent on a fully-loaded beatmap.</param>
2017-05-19 07:39:02 +00:00
/// <returns>The hit objects generated.</returns>
2018-04-19 11:44:38 +00:00
private IEnumerable<ManiaHitObject> generateSpecific(HitObject original, IBeatmap originalBeatmap)
2017-05-19 07:31:05 +00:00
{
var generator = new SpecificBeatmapPatternGenerator(Random, original, originalBeatmap, TotalColumns, lastPattern);
2018-04-13 09:19:50 +00:00
foreach (var newPattern in generator.Generate())
{
lastPattern = newPattern;
2018-04-13 09:19:50 +00:00
foreach (var obj in newPattern.HitObjects)
yield return obj;
}
2017-05-19 07:31:05 +00:00
}
2018-04-13 09:19:50 +00:00
2017-05-19 07:39:02 +00:00
/// <summary>
/// Method that generates hit objects for non-osu!mania beatmaps.
/// </summary>
/// <param name="original">The original hit object.</param>
/// <param name="originalBeatmap">The original beatmap. This is used to look-up any values dependent on a fully-loaded beatmap.</param>
2017-05-19 07:39:02 +00:00
/// <returns>The hit objects generated.</returns>
2018-04-19 11:44:38 +00:00
private IEnumerable<ManiaHitObject> generateConverted(HitObject original, IBeatmap originalBeatmap)
2017-05-19 07:31:05 +00:00
{
Patterns.PatternGenerator? conversion = null;
2018-04-13 09:19:50 +00:00
2019-11-12 10:16:51 +00:00
switch (original)
2018-06-15 11:52:09 +00:00
{
case IHasPath:
2019-11-12 10:16:51 +00:00
{
var generator = new PathObjectPatternGenerator(Random, original, originalBeatmap, TotalColumns, lastPattern);
2019-11-12 10:16:51 +00:00
conversion = generator;
var positionData = original as IHasPosition;
for (int i = 0; i <= generator.SpanCount; i++)
2019-11-12 10:16:51 +00:00
{
double time = original.StartTime + generator.SegmentDuration * i;
2019-11-12 10:16:51 +00:00
recordNote(time, positionData?.Position ?? Vector2.Zero);
computeDensity(time);
}
2018-06-15 11:52:36 +00:00
2019-11-12 10:16:51 +00:00
break;
}
2020-05-27 03:38:39 +00:00
case IHasDuration endTimeData:
2018-06-15 11:52:36 +00:00
{
conversion = new EndTimeObjectPatternGenerator(Random, original, originalBeatmap, TotalColumns, lastPattern);
2019-11-12 10:16:51 +00:00
recordNote(endTimeData.EndTime, new Vector2(256, 192));
computeDensity(endTimeData.EndTime);
break;
2018-06-15 11:52:36 +00:00
}
2019-11-12 10:16:51 +00:00
case IHasPosition positionData:
{
computeDensity(original.StartTime);
2018-04-13 09:19:50 +00:00
conversion = new HitObjectPatternGenerator(Random, original, originalBeatmap, TotalColumns, lastPattern, lastTime, lastPosition, density, lastStair);
2018-04-13 09:19:50 +00:00
2019-11-12 10:16:51 +00:00
recordNote(original.StartTime, positionData.Position);
break;
}
2017-05-19 07:31:05 +00:00
}
2018-04-13 09:19:50 +00:00
if (conversion == null)
yield break;
2018-04-13 09:19:50 +00:00
foreach (var newPattern in conversion.Generate())
{
lastPattern = conversion is EndTimeObjectPatternGenerator ? lastPattern : newPattern;
lastStair = (conversion as HitObjectPatternGenerator)?.StairType ?? lastStair;
2018-04-13 09:19:50 +00:00
foreach (var obj in newPattern.HitObjects)
yield return obj;
}
2017-05-19 07:31:05 +00:00
}
2018-04-13 09:19:50 +00:00
2017-05-19 07:31:05 +00:00
/// <summary>
2017-05-19 07:39:02 +00:00
/// A pattern generator for osu!mania-specific beatmaps.
2017-05-19 07:31:05 +00:00
/// </summary>
private class SpecificBeatmapPatternGenerator : Patterns.Legacy.PatternGenerator
{
public SpecificBeatmapPatternGenerator(LegacyRandom random, HitObject hitObject, IBeatmap beatmap, int totalColumns, Pattern previousPattern)
: base(random, hitObject, beatmap, previousPattern, totalColumns)
2017-05-19 07:31:05 +00:00
{
}
2018-04-13 09:19:50 +00:00
public override IEnumerable<Pattern> Generate()
{
yield return generate();
}
private Pattern generate()
2017-05-19 07:31:05 +00:00
{
var positionData = HitObject as IHasXPosition;
2018-04-13 09:19:50 +00:00
2017-05-19 07:31:05 +00:00
int column = GetColumn(positionData?.X ?? 0);
2018-04-13 09:19:50 +00:00
2017-05-19 07:31:05 +00:00
var pattern = new Pattern();
2018-04-13 09:19:50 +00:00
2020-05-27 03:38:39 +00:00
if (HitObject is IHasDuration endTimeData)
2017-05-19 07:31:05 +00:00
{
pattern.Add(new HoldNote
{
StartTime = HitObject.StartTime,
Duration = endTimeData.Duration,
2020-04-22 04:45:12 +00:00
Column = column,
2020-04-21 07:33:19 +00:00
Samples = HitObject.Samples,
NodeSamples = (HitObject as IHasRepeats)?.NodeSamples ?? HoldNote.CreateDefaultNodeSamples(HitObject)
2017-05-19 07:31:05 +00:00
});
}
2019-11-12 10:16:51 +00:00
else if (HitObject is IHasXPosition)
2017-05-19 07:31:05 +00:00
{
pattern.Add(new Note
{
StartTime = HitObject.StartTime,
Samples = HitObject.Samples,
Column = column
});
}
2018-04-13 09:19:50 +00:00
2017-05-19 07:31:05 +00:00
return pattern;
}
}
2017-03-11 15:34:21 +00:00
}
}