2019-09-10 12:29:50 +08: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.
|
|
|
|
|
2020-05-18 04:08:49 +08:00
|
|
|
using System;
|
2019-09-10 12:29:50 +08:00
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Linq;
|
2020-01-09 12:43:44 +08:00
|
|
|
using osu.Framework.Utils;
|
2019-09-10 12:29:50 +08:00
|
|
|
using osu.Game.Beatmaps;
|
|
|
|
using osu.Game.Beatmaps.ControlPoints;
|
|
|
|
|
|
|
|
namespace osu.Game.Rulesets.Objects
|
|
|
|
{
|
2019-09-25 05:03:55 +08:00
|
|
|
public class BarLineGenerator<TBarLine>
|
|
|
|
where TBarLine : class, IBarLine, new()
|
2019-09-10 12:29:50 +08:00
|
|
|
{
|
|
|
|
/// <summary>
|
|
|
|
/// The generated bar lines.
|
|
|
|
/// </summary>
|
2019-09-25 05:03:55 +08:00
|
|
|
public readonly List<TBarLine> BarLines = new List<TBarLine>();
|
2019-09-10 12:29:50 +08:00
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Constructs and generates bar lines for provided beatmap.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="beatmap">The beatmap to generate bar lines for.</param>
|
|
|
|
public BarLineGenerator(IBeatmap beatmap)
|
|
|
|
{
|
|
|
|
if (beatmap.HitObjects.Count == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
HitObject lastObject = beatmap.HitObjects.Last();
|
2019-11-25 21:37:07 +08:00
|
|
|
double lastHitTime = 1 + lastObject.GetEndTime();
|
2019-09-10 12:29:50 +08:00
|
|
|
|
|
|
|
var timingPoints = beatmap.ControlPointInfo.TimingPoints;
|
2022-10-08 02:38:32 +08:00
|
|
|
var effectPoints = beatmap.ControlPointInfo.EffectPoints;
|
2019-09-10 12:29:50 +08:00
|
|
|
|
|
|
|
if (timingPoints.Count == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (int i = 0; i < timingPoints.Count; i++)
|
|
|
|
{
|
|
|
|
TimingControlPoint currentTimingPoint = timingPoints[i];
|
2022-10-08 02:38:32 +08:00
|
|
|
EffectControlPoint? currentEffectPoint = effectPoints.FirstOrDefault(p => p.Time == currentTimingPoint.Time);
|
2019-09-10 12:29:50 +08:00
|
|
|
int currentBeat = 0;
|
|
|
|
|
|
|
|
// Stop on the beat before the next timing point, or if there is no next timing point stop slightly past the last object
|
2022-10-08 02:34:31 +08:00
|
|
|
double endTime = i < timingPoints.Count - 1 ? timingPoints[i + 1].Time : lastHitTime + currentTimingPoint.BeatLength * currentTimingPoint.TimeSignature.Numerator;
|
2019-09-10 12:29:50 +08:00
|
|
|
|
2022-10-08 02:38:32 +08:00
|
|
|
double startTime = currentTimingPoint.Time;
|
|
|
|
|
2022-01-23 00:27:27 +08:00
|
|
|
double barLength = currentTimingPoint.BeatLength * currentTimingPoint.TimeSignature.Numerator;
|
2019-09-10 12:29:50 +08:00
|
|
|
|
2022-10-08 02:38:32 +08:00
|
|
|
if (currentEffectPoint != null && currentEffectPoint.OmitFirstBarLine)
|
|
|
|
{
|
|
|
|
startTime += barLength;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (double t = startTime; Precision.DefinitelyBigger(endTime, t); t += barLength, currentBeat++)
|
2019-09-10 12:29:50 +08:00
|
|
|
{
|
2021-10-27 12:04:41 +08:00
|
|
|
double roundedTime = Math.Round(t, MidpointRounding.AwayFromZero);
|
2020-05-18 04:08:49 +08:00
|
|
|
|
|
|
|
// in the case of some bar lengths, rounding errors can cause t to be slightly less than
|
|
|
|
// the expected whole number value due to floating point inaccuracies.
|
|
|
|
// if this is the case, apply rounding.
|
|
|
|
if (Precision.AlmostEquals(t, roundedTime))
|
|
|
|
{
|
|
|
|
t = roundedTime;
|
|
|
|
}
|
|
|
|
|
2019-09-25 05:03:55 +08:00
|
|
|
BarLines.Add(new TBarLine
|
2019-09-10 12:29:50 +08:00
|
|
|
{
|
|
|
|
StartTime = t,
|
2022-01-23 00:27:27 +08:00
|
|
|
Major = currentBeat % currentTimingPoint.TimeSignature.Numerator == 0
|
2019-09-10 12:29:50 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|