2021-04-25 03:34:54 +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.
|
|
|
|
|
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using osu.Game.Beatmaps;
|
|
|
|
using osu.Game.Rulesets.Edit.Checks.Components;
|
|
|
|
using osu.Game.Rulesets.Objects;
|
|
|
|
using osu.Game.Rulesets.Objects.Types;
|
|
|
|
|
|
|
|
namespace osu.Game.Rulesets.Edit.Checks
|
|
|
|
{
|
2021-04-26 18:32:44 +00:00
|
|
|
public class CheckUnsnappedObjects : ICheck
|
2021-04-25 03:34:54 +00:00
|
|
|
{
|
2021-04-26 18:17:18 +00:00
|
|
|
public const double UNSNAP_MS_THRESHOLD = 2;
|
2021-04-25 03:34:54 +00:00
|
|
|
|
2021-04-27 00:32:57 +00:00
|
|
|
public CheckMetadata Metadata { get; } = new CheckMetadata(CheckCategory.Timing, "Unsnapped hitobjects");
|
2021-04-25 03:34:54 +00:00
|
|
|
|
|
|
|
public IEnumerable<IssueTemplate> PossibleTemplates => new IssueTemplate[]
|
|
|
|
{
|
2021-04-27 11:56:05 +00:00
|
|
|
new IssueTemplateLargeUnsnap(this),
|
|
|
|
new IssueTemplateSmallUnsnap(this)
|
2021-04-25 03:34:54 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
public IEnumerable<Issue> Run(IBeatmap playableBeatmap, IWorkingBeatmap workingBeatmap)
|
|
|
|
{
|
2021-04-28 07:47:30 +00:00
|
|
|
var controlPointInfo = playableBeatmap.ControlPointInfo;
|
|
|
|
|
2021-04-25 03:34:54 +00:00
|
|
|
foreach (var hitobject in playableBeatmap.HitObjects)
|
|
|
|
{
|
2021-04-28 07:47:30 +00:00
|
|
|
double startUnsnap = hitobject.StartTime - controlPointInfo.ClosestSnapTime(hitobject.StartTime);
|
2021-04-25 03:34:54 +00:00
|
|
|
string startPostfix = hitobject is IHasDuration ? "start" : "";
|
|
|
|
foreach (var issue in getUnsnapIssues(hitobject, startUnsnap, hitobject.StartTime, startPostfix))
|
|
|
|
yield return issue;
|
|
|
|
|
|
|
|
if (hitobject is IHasRepeats hasRepeats)
|
|
|
|
{
|
|
|
|
for (int repeatIndex = 0; repeatIndex < hasRepeats.RepeatCount; ++repeatIndex)
|
|
|
|
{
|
|
|
|
double spanDuration = hasRepeats.Duration / (hasRepeats.RepeatCount + 1);
|
|
|
|
double repeatTime = hitobject.StartTime + spanDuration * (repeatIndex + 1);
|
2021-04-28 07:47:30 +00:00
|
|
|
double repeatUnsnap = repeatTime - controlPointInfo.ClosestSnapTime(repeatTime);
|
2021-04-25 03:34:54 +00:00
|
|
|
foreach (var issue in getUnsnapIssues(hitobject, repeatUnsnap, repeatTime, "repeat"))
|
|
|
|
yield return issue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hitobject is IHasDuration hasDuration)
|
|
|
|
{
|
2021-04-28 07:47:30 +00:00
|
|
|
double endUnsnap = hasDuration.EndTime - controlPointInfo.ClosestSnapTime(hasDuration.EndTime);
|
2021-04-25 03:34:54 +00:00
|
|
|
foreach (var issue in getUnsnapIssues(hitobject, endUnsnap, hasDuration.EndTime, "end"))
|
|
|
|
yield return issue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private IEnumerable<Issue> getUnsnapIssues(HitObject hitobject, double unsnap, double time, string postfix = "")
|
|
|
|
{
|
2021-04-26 18:17:18 +00:00
|
|
|
if (Math.Abs(unsnap) >= UNSNAP_MS_THRESHOLD)
|
2021-04-27 11:56:05 +00:00
|
|
|
yield return new IssueTemplateLargeUnsnap(this).Create(hitobject, unsnap, time, postfix);
|
2021-04-25 03:34:54 +00:00
|
|
|
else if (Math.Abs(unsnap) >= 1)
|
2021-04-27 11:56:05 +00:00
|
|
|
yield return new IssueTemplateSmallUnsnap(this).Create(hitobject, unsnap, time, postfix);
|
2021-04-25 03:34:54 +00:00
|
|
|
|
|
|
|
// We don't care about unsnaps < 1 ms, as all object ends have these due to the way SV works.
|
|
|
|
}
|
|
|
|
|
|
|
|
public abstract class IssueTemplateUnsnap : IssueTemplate
|
|
|
|
{
|
|
|
|
protected IssueTemplateUnsnap(ICheck check, IssueType type)
|
2021-04-26 14:17:38 +00:00
|
|
|
: base(check, type, "{0} is unsnapped by {1:0.##} ms.")
|
2021-04-25 03:34:54 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
public Issue Create(HitObject hitobject, double unsnap, double time, string postfix = "")
|
|
|
|
{
|
|
|
|
string objectName = hitobject.GetType().Name;
|
|
|
|
if (!string.IsNullOrEmpty(postfix))
|
|
|
|
objectName += " " + postfix;
|
|
|
|
|
|
|
|
return new Issue(hitobject, this, objectName, unsnap) { Time = time };
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-27 11:56:05 +00:00
|
|
|
public class IssueTemplateLargeUnsnap : IssueTemplateUnsnap
|
2021-04-25 03:34:54 +00:00
|
|
|
{
|
2021-04-27 11:56:05 +00:00
|
|
|
public IssueTemplateLargeUnsnap(ICheck check)
|
2021-04-25 03:34:54 +00:00
|
|
|
: base(check, IssueType.Problem)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-27 11:56:05 +00:00
|
|
|
public class IssueTemplateSmallUnsnap : IssueTemplateUnsnap
|
2021-04-25 03:34:54 +00:00
|
|
|
{
|
2021-04-27 11:56:05 +00:00
|
|
|
public IssueTemplateSmallUnsnap(ICheck check)
|
2021-04-25 03:34:54 +00:00
|
|
|
: base(check, IssueType.Negligible)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|