// Copyright (c) ppy Pty Ltd . Licensed under the MIT Licence. // See the LICENCE file in the repository root for full licence text. using NUnit.Framework; using osu.Framework.Allocation; using osu.Framework.Graphics; using osu.Framework.Testing; using osu.Game.Rulesets.Judgements; using osu.Game.Rulesets.Osu.Objects; using osu.Game.Rulesets.Osu.Scoring; using osu.Game.Rulesets.Scoring; using osu.Game.Screens.Play.HUD; using osuTK; namespace osu.Game.Tests.Visual.Gameplay { public class TestSceneUnstableRateCounter : OsuTestScene { [Cached(typeof(ScoreProcessor))] private TestScoreProcessor scoreProcessor = new TestScoreProcessor(); private readonly OsuHitWindows hitWindows = new OsuHitWindows(); private UnstableRateCounter counter; private double prev; [SetUpSteps] public void SetUp() { AddStep("Reset Score Processor", () => scoreProcessor.Reset()); } [Test] public void TestBasic() { AddStep("Create Display", recreateDisplay); // Needs multiples 2 by the nature of UR, and went for 4 to be safe. // Creates a 250 UR by placing a +25ms then a -25ms judgement, which then results in a 250 UR AddRepeatStep("Set UR to 250", () => applyJudgement(25, true), 4); AddUntilStep("UR = 250", () => counter.Current.Value == 250.0); AddRepeatStep("Revert UR", () => { scoreProcessor.RevertResult( new JudgementResult(new HitCircle { HitWindows = hitWindows }, new Judgement()) { TimeOffset = 25, Type = HitResult.Perfect, }); }, 4); AddUntilStep("UR is 0", () => counter.Current.Value == 0.0); AddUntilStep("Counter is invalid", () => counter.Child.Alpha == 0.3f); //Sets a UR of 0 by creating 10 10ms offset judgements. Since average = offset, UR = 0 AddRepeatStep("Set UR to 0", () => applyJudgement(10, false), 10); //Applies a UR of 100 by creating 10 -10ms offset judgements. At the 10th judgement, offset should be 100. AddRepeatStep("Bring UR to 100", () => applyJudgement(-10, false), 10); } private void recreateDisplay() { Clear(); Add(counter = new UnstableRateCounter { Anchor = Anchor.Centre, Origin = Anchor.Centre, Scale = new Vector2(5), }); } private void applyJudgement(double offsetMs, bool alt) { double placement = offsetMs; if (alt) { placement = prev > 0 ? -offsetMs : offsetMs; prev = placement; } scoreProcessor.ApplyResult(new JudgementResult(new HitCircle { HitWindows = hitWindows }, new Judgement()) { TimeOffset = placement, Type = HitResult.Perfect, }); } private class TestScoreProcessor : ScoreProcessor { public void Reset() => base.Reset(false); } } }