// Copyright (c) 2007-2017 ppy Pty Ltd . // Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE using System.Collections.Generic; using System.Linq; namespace osu.Game.Beatmaps.Timing { public class TimingInfo { public readonly List ControlPoints = new List(); public double BPMMaximum => 60000 / (ControlPoints?.Where(c => c.BeatLength != 0).OrderBy(c => c.BeatLength).FirstOrDefault() ?? ControlPoint.Default).BeatLength; public double BPMMinimum => 60000 / (ControlPoints?.Where(c => c.BeatLength != 0).OrderByDescending(c => c.BeatLength).FirstOrDefault() ?? ControlPoint.Default).BeatLength; public double BPMMode => BPMAt(ControlPoints.Where(c => c.BeatLength != 0).GroupBy(c => c.BeatLength).OrderByDescending(grp => grp.Count()).First().First().Time); public double BPMAt(double time) { return 60000 / BeatLengthAt(time); } /// /// Finds the speed multiplier at a time. /// /// The time to find the speed multiplier at. /// The speed multiplier. public double SpeedMultiplierAt(double time) { ControlPoint overridePoint; ControlPoint timingPoint = TimingPointAt(time, out overridePoint); return overridePoint?.VelocityAdjustment ?? timingPoint?.VelocityAdjustment ?? 1; } /// /// Finds the beat length at a time. This is expressed in milliseconds. /// /// The time to find the beat length at. /// The beat length. public double BeatLengthAt(double time) { ControlPoint overridePoint; ControlPoint timingPoint = TimingPointAt(time, out overridePoint); return timingPoint.BeatLength; } /// /// Finds the timing point at a time. /// /// The time to find the timing point at. /// The timing point containing the velocity change of the returned timing point. /// The timing point. public ControlPoint TimingPointAt(double time, out ControlPoint overridePoint) { overridePoint = null; ControlPoint timingPoint = null; foreach (var controlPoint in ControlPoints) { // Some beatmaps have the first timingPoint (accidentally) start after the first HitObject(s). // This null check makes it so that the first ControlPoint that makes a timing change is used as // the timingPoint for those HitObject(s). if (controlPoint.Time <= time || timingPoint == null) { if (controlPoint.TimingChange) { timingPoint = controlPoint; overridePoint = null; } else overridePoint = controlPoint; } else break; } return timingPoint ?? ControlPoint.Default; } } }