2019-01-24 08:43:03 +00:00
|
|
|
|
// 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
|
|
|
|
|
2022-06-17 07:37:17 +00:00
|
|
|
|
#nullable disable
|
|
|
|
|
|
2017-04-18 07:05:58 +00:00
|
|
|
|
using osu.Game.Rulesets.Mania.Objects;
|
2017-04-17 06:44:46 +00:00
|
|
|
|
using System;
|
2018-01-03 09:44:25 +00:00
|
|
|
|
using System.Linq;
|
2017-05-17 04:07:56 +00:00
|
|
|
|
using System.Collections.Generic;
|
2020-09-17 08:40:05 +00:00
|
|
|
|
using System.Threading;
|
2020-06-04 20:28:55 +00:00
|
|
|
|
using osu.Game.Audio;
|
2017-05-17 04:07:56 +00:00
|
|
|
|
using osu.Game.Beatmaps;
|
2017-04-18 07:05:58 +00:00
|
|
|
|
using osu.Game.Rulesets.Objects;
|
2017-05-17 04:07:56 +00:00
|
|
|
|
using osu.Game.Rulesets.Objects.Types;
|
2017-05-19 07:31:05 +00:00
|
|
|
|
using osu.Game.Rulesets.Mania.Beatmaps.Patterns;
|
2017-05-19 11:57:20 +00:00
|
|
|
|
using osu.Game.Rulesets.Mania.Beatmaps.Patterns.Legacy;
|
2023-10-02 06:53:05 +00:00
|
|
|
|
using osu.Game.Rulesets.Scoring.Legacy;
|
2022-04-28 08:46:00 +00:00
|
|
|
|
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
|
|
|
|
{
|
2017-05-17 04:07:56 +00:00
|
|
|
|
public class ManiaBeatmapConverter : BeatmapConverter<ManiaHitObject>
|
2017-03-11 15:34:21 +00:00
|
|
|
|
{
|
2017-05-19 11:57:20 +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
|
|
|
|
|
2018-01-03 09:44:25 +00:00
|
|
|
|
public int TargetColumns;
|
2019-02-28 14:40:03 +00:00
|
|
|
|
public bool Dual;
|
2018-01-03 09:44:25 +00:00
|
|
|
|
public readonly bool IsForCurrentRuleset;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2020-10-14 11:40:17 +00:00
|
|
|
|
private readonly int originalTargetColumns;
|
2020-10-14 08:53:28 +00:00
|
|
|
|
|
2018-06-15 11:52:09 +00:00
|
|
|
|
// Internal for testing purposes
|
2022-04-28 08:46:00 +00:00
|
|
|
|
internal LegacyRandom Random { get; private set; }
|
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
|
|
|
|
|
2018-01-03 09:44:25 +00:00
|
|
|
|
private ManiaBeatmap beatmap;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2019-12-24 07:02:16 +00:00
|
|
|
|
public ManiaBeatmapConverter(IBeatmap beatmap, Ruleset ruleset)
|
|
|
|
|
: base(beatmap, ruleset)
|
2017-03-11 15:34:21 +00:00
|
|
|
|
{
|
2019-12-24 07:02:16 +00:00
|
|
|
|
IsForCurrentRuleset = beatmap.BeatmapInfo.Ruleset.Equals(ruleset.RulesetInfo);
|
2023-10-02 06:53:05 +00:00
|
|
|
|
TargetColumns = GetColumnCount(LegacyBeatmapConversionDifficultyInfo.FromBeatmap(beatmap));
|
2018-05-07 02:23:29 +00:00
|
|
|
|
|
2023-10-02 06:53:05 +00:00
|
|
|
|
if (IsForCurrentRuleset && TargetColumns > ManiaRuleset.MAX_STAGE_KEYS)
|
2018-01-03 09:44:25 +00:00
|
|
|
|
{
|
2023-10-02 06:53:05 +00:00
|
|
|
|
TargetColumns /= 2;
|
|
|
|
|
Dual = true;
|
2018-01-03 09:44:25 +00:00
|
|
|
|
}
|
2020-10-14 08:53:28 +00:00
|
|
|
|
|
|
|
|
|
originalTargetColumns = TargetColumns;
|
2017-08-22 04:01:51 +00:00
|
|
|
|
}
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2023-10-02 06:53:05 +00:00
|
|
|
|
public static int GetColumnCount(LegacyBeatmapConversionDifficultyInfo difficulty)
|
|
|
|
|
{
|
2023-12-09 13:09:49 +00:00
|
|
|
|
double roundedCircleSize = Math.Round(difficulty.CircleSize);
|
|
|
|
|
|
2023-12-19 09:27:44 +00:00
|
|
|
|
if (difficulty.SourceRuleset.ShortName == ManiaRuleset.SHORT_NAME)
|
2023-12-09 13:09:49 +00:00
|
|
|
|
return (int)Math.Max(1, roundedCircleSize);
|
2023-10-02 06:53:05 +00:00
|
|
|
|
|
|
|
|
|
double roundedOverallDifficulty = Math.Round(difficulty.OverallDifficulty);
|
|
|
|
|
|
2023-12-19 09:27:44 +00:00
|
|
|
|
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;
|
|
|
|
|
}
|
2023-10-02 06:53:05 +00:00
|
|
|
|
|
|
|
|
|
return Math.Max(4, Math.Min((int)roundedOverallDifficulty + 1, 7));
|
|
|
|
|
}
|
|
|
|
|
|
2020-04-21 07:44:26 +00:00
|
|
|
|
public override bool CanConvert() => Beatmap.HitObjects.All(h => h is IHasXPosition);
|
2019-12-23 08:44:18 +00:00
|
|
|
|
|
2020-09-17 08:40:05 +00:00
|
|
|
|
protected override Beatmap<ManiaHitObject> ConvertBeatmap(IBeatmap original, CancellationToken cancellationToken)
|
2017-08-22 04:01:51 +00:00
|
|
|
|
{
|
2021-10-02 03:34:29 +00:00
|
|
|
|
IBeatmapDifficultyInfo difficulty = original.Difficulty;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2019-11-24 23:45:42 +00:00
|
|
|
|
int seed = (int)MathF.Round(difficulty.DrainRate + difficulty.CircleSize) * 20 + (int)(difficulty.OverallDifficulty * 41.2) + (int)MathF.Round(difficulty.ApproachRate);
|
2022-04-28 08:46:00 +00:00
|
|
|
|
Random = new LegacyRandom(seed);
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2020-09-17 08:40:05 +00:00
|
|
|
|
return base.ConvertBeatmap(original, cancellationToken);
|
2017-03-11 15:34:21 +00:00
|
|
|
|
}
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2019-02-28 10:07:43 +00:00
|
|
|
|
protected override Beatmap<ManiaHitObject> CreateBeatmap()
|
|
|
|
|
{
|
2022-10-05 10:14:31 +00:00
|
|
|
|
beatmap = new ManiaBeatmap(new StageDefinition(TargetColumns), originalTargetColumns);
|
2019-02-28 10:07:43 +00:00
|
|
|
|
|
2019-03-01 05:30:58 +00:00
|
|
|
|
if (Dual)
|
2022-10-05 10:14:31 +00:00
|
|
|
|
beatmap.Stages.Add(new StageDefinition(TargetColumns));
|
2019-02-28 10:07:43 +00:00
|
|
|
|
|
|
|
|
|
return beatmap;
|
|
|
|
|
}
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2020-09-17 08:40:05 +00:00
|
|
|
|
protected override IEnumerable<ManiaHitObject> ConvertHitObject(HitObject original, IBeatmap beatmap, CancellationToken cancellationToken)
|
2017-05-17 04:07:56 +00:00
|
|
|
|
{
|
2019-02-28 05:35:00 +00:00
|
|
|
|
if (original is ManiaHitObject maniaOriginal)
|
2017-05-19 06:57:32 +00:00
|
|
|
|
{
|
|
|
|
|
yield return maniaOriginal;
|
2019-02-28 04:31:40 +00:00
|
|
|
|
|
2017-05-19 06:57:32 +00:00
|
|
|
|
yield break;
|
|
|
|
|
}
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2018-03-14 02:27:14 +00:00
|
|
|
|
var objects = IsForCurrentRuleset ? generateSpecific(original, beatmap) : generateConverted(original, beatmap);
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2017-05-19 07:31:05 +00:00
|
|
|
|
if (objects == null)
|
|
|
|
|
yield break;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2017-05-19 07:31:05 +00:00
|
|
|
|
foreach (ManiaHitObject obj in objects)
|
|
|
|
|
yield return obj;
|
|
|
|
|
}
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2023-06-15 11:00:15 +00:00
|
|
|
|
private readonly LimitedCapacityQueue<double> prevNoteTimes = new LimitedCapacityQueue<double>(max_notes_for_density);
|
2017-05-19 11:57:20 +00:00
|
|
|
|
private double density = int.MaxValue;
|
2019-02-28 04:31:40 +00:00
|
|
|
|
|
2017-05-19 11:57:20 +00:00
|
|
|
|
private void computeDensity(double newNoteTime)
|
|
|
|
|
{
|
2023-06-15 11:00:15 +00:00
|
|
|
|
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;
|
2017-05-19 11:57:20 +00:00
|
|
|
|
}
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2017-05-19 11:57:20 +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
|
|
|
|
|
2017-05-19 11:57:20 +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>
|
2018-03-14 02:27:14 +00:00
|
|
|
|
/// <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
|
|
|
|
{
|
2018-06-15 11:52:09 +00:00
|
|
|
|
var generator = new SpecificBeatmapPatternGenerator(Random, original, beatmap, lastPattern, originalBeatmap);
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2018-06-15 13:10:57 +00:00
|
|
|
|
foreach (var newPattern in generator.Generate())
|
|
|
|
|
{
|
|
|
|
|
lastPattern = newPattern;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2018-06-15 13:10:57 +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>
|
2018-03-14 02:27:14 +00:00
|
|
|
|
/// <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
|
|
|
|
{
|
2017-05-19 11:57:20 +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
|
|
|
|
{
|
2023-11-29 08:30:21 +00:00
|
|
|
|
case IHasPath:
|
2019-11-12 10:16:51 +00:00
|
|
|
|
{
|
2023-11-29 08:30:21 +00:00
|
|
|
|
var generator = new PathObjectPatternGenerator(Random, original, beatmap, lastPattern, originalBeatmap);
|
2019-11-12 10:16:51 +00:00
|
|
|
|
conversion = generator;
|
|
|
|
|
|
|
|
|
|
var positionData = original as IHasPosition;
|
|
|
|
|
|
2020-09-07 07:30:05 +00:00
|
|
|
|
for (int i = 0; i <= generator.SpanCount; i++)
|
2019-11-12 10:16:51 +00:00
|
|
|
|
{
|
2020-09-07 07:30:05 +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
|
|
|
|
{
|
2020-10-09 12:12:38 +00:00
|
|
|
|
conversion = new EndTimeObjectPatternGenerator(Random, original, beatmap, lastPattern, originalBeatmap);
|
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
|
|
|
|
|
2019-11-12 10:16:51 +00:00
|
|
|
|
conversion = new HitObjectPatternGenerator(Random, original, beatmap, lastPattern, lastTime, lastPosition, density, lastStair, originalBeatmap);
|
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
|
|
|
|
|
2017-05-19 11:57:20 +00:00
|
|
|
|
if (conversion == null)
|
2018-06-15 13:10:57 +00:00
|
|
|
|
yield break;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2018-06-15 13:10:57 +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
|
|
|
|
|
2018-06-15 13:10:57 +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
|
|
|
|
|
{
|
2022-04-28 08:46:00 +00:00
|
|
|
|
public SpecificBeatmapPatternGenerator(LegacyRandom random, HitObject hitObject, ManiaBeatmap beatmap, Pattern previousPattern, IBeatmap originalBeatmap)
|
2018-03-14 02:27:14 +00:00
|
|
|
|
: base(random, hitObject, beatmap, previousPattern, originalBeatmap)
|
2017-05-19 07:31:05 +00:00
|
|
|
|
{
|
|
|
|
|
}
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2018-06-15 13:10:57 +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,
|
2020-06-04 20:28:55 +00:00
|
|
|
|
NodeSamples = (HitObject as IHasRepeats)?.NodeSamples ?? defaultNodeSamples
|
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;
|
|
|
|
|
}
|
2020-06-04 20:28:55 +00:00
|
|
|
|
|
|
|
|
|
/// <remarks>
|
|
|
|
|
/// osu!mania-specific beatmaps in stable only play samples at the start of the hold note.
|
|
|
|
|
/// </remarks>
|
|
|
|
|
private List<IList<HitSampleInfo>> defaultNodeSamples
|
|
|
|
|
=> new List<IList<HitSampleInfo>>
|
|
|
|
|
{
|
|
|
|
|
HitObject.Samples,
|
|
|
|
|
new List<HitSampleInfo>()
|
|
|
|
|
};
|
2017-05-17 04:07:56 +00:00
|
|
|
|
}
|
2017-03-11 15:34:21 +00:00
|
|
|
|
}
|
|
|
|
|
}
|