// Copyright (c) ppy Pty Ltd . Licensed under the MIT Licence. // See the LICENCE file in the repository root for full licence text. using System.Collections.Generic; using System.Linq; using osu.Framework.Extensions.ObjectExtensions; using osu.Game.Rulesets.Mania.Beatmaps; using osu.Game.Rulesets.Mania.Objects; using osu.Game.Rulesets.Objects; using osu.Game.Rulesets.Replays; namespace osu.Game.Rulesets.Mania.Replays { internal class ManiaAutoGenerator : AutoGenerator { public const double RELEASE_DELAY = 20; public new ManiaBeatmap Beatmap => (ManiaBeatmap)base.Beatmap; public ManiaAutoGenerator(ManiaBeatmap beatmap) : base(beatmap) { } protected override void GenerateFrames() { if (Beatmap.HitObjects.Count == 0) return; var pointGroups = generateActionPoints().GroupBy(a => a.Time).OrderBy(g => g.First().Time); var actions = new List(); foreach (var group in pointGroups) { foreach (var point in group) { switch (point) { case HitPoint: actions.Add(ManiaAction.Key1 + point.Column); break; case ReleasePoint: actions.Remove(ManiaAction.Key1 + point.Column); break; } } Frames.Add(new ManiaReplayFrame(group.First().Time, actions.ToArray())); } } private IEnumerable generateActionPoints() { for (int i = 0; i < Beatmap.HitObjects.Count; i++) { var currentObject = Beatmap.HitObjects[i]; var nextObjectInColumn = GetNextObject(i); // Get the next object that requires pressing the same button double releaseTime = calculateReleaseTime(currentObject, nextObjectInColumn); yield return new HitPoint { Time = currentObject.StartTime, Column = currentObject.Column }; yield return new ReleasePoint { Time = releaseTime, Column = currentObject.Column }; } } private double calculateReleaseTime(HitObject currentObject, HitObject? nextObject) { double endTime = currentObject.GetEndTime(); double releaseDelay = RELEASE_DELAY; if (currentObject is HoldNote hold) { if (hold.Duration > 0) // hold note releases must be timed exactly. return endTime; // Special case for super short hold notes releaseDelay = 1; } bool canDelayKeyUpFully = nextObject == null || nextObject.StartTime > endTime + releaseDelay; return endTime + (canDelayKeyUpFully ? releaseDelay : (nextObject.AsNonNull().StartTime - endTime) * 0.9); } protected override HitObject? GetNextObject(int currentIndex) { int desiredColumn = Beatmap.HitObjects[currentIndex].Column; for (int i = currentIndex + 1; i < Beatmap.HitObjects.Count; i++) { if (Beatmap.HitObjects[i].Column == desiredColumn) return Beatmap.HitObjects[i]; } return null; } private interface IActionPoint { double Time { get; set; } int Column { get; set; } } private struct HitPoint : IActionPoint { public double Time { get; set; } public int Column { get; set; } } private struct ReleasePoint : IActionPoint { public double Time { get; set; } public int Column { get; set; } } } }