// Copyright (c) ppy Pty Ltd . Licensed under the MIT Licence. // See the LICENCE file in the repository root for full licence text. using System; using System.Collections.Generic; using System.Linq; using NUnit.Framework; using osu.Framework.Testing; using osu.Framework.Utils; using osu.Game.Beatmaps; using osu.Game.Rulesets.Judgements; using osu.Game.Rulesets.Mods; using osu.Game.Rulesets.Objects; using osu.Game.Rulesets.Osu.Judgements; using osu.Game.Rulesets.Osu.Mods; using osu.Game.Rulesets.Osu.Objects; using osu.Game.Rulesets.Osu.Objects.Drawables; using osu.Game.Rulesets.Osu.Skinning.Default; using osu.Game.Rulesets.Scoring; using osuTK; namespace osu.Game.Rulesets.Osu.Tests.Mods { public class TestSceneOsuModSpunOut : OsuModTestScene { protected override bool AllowFail => true; [Test] public void TestSpinnerAutoCompleted() { DrawableSpinner? spinner = null; JudgementResult? lastResult = null; CreateModTest(new ModTestData { Mod = new OsuModSpunOut(), Autoplay = false, Beatmap = singleSpinnerBeatmap, PassCondition = () => { // Bind to the first spinner's results for further tracking. if (spinner == null) { // We only care about the first spinner we encounter for this test. var nextSpinner = Player.ChildrenOfType().SingleOrDefault(); if (nextSpinner == null) return false; lastResult = null; spinner = nextSpinner; spinner.OnNewResult += (_, result) => lastResult = result; } return lastResult?.Type == HitResult.Great; } }); } [TestCase(null)] [TestCase(typeof(OsuModDoubleTime))] [TestCase(typeof(OsuModHalfTime))] public void TestSpinRateUnaffectedByMods(Type? additionalModType) { var mods = new List { new OsuModSpunOut() }; if (additionalModType != null) mods.Add((Mod)Activator.CreateInstance(additionalModType)!); CreateModTest(new ModTestData { Mods = mods, Autoplay = false, Beatmap = singleSpinnerBeatmap, PassCondition = () => { var counter = Player.ChildrenOfType().SingleOrDefault(); var spinner = Player.ChildrenOfType().FirstOrDefault(); if (counter == null || spinner == null) return false; // ignore cases where the spinner hasn't started as these lead to false-positives if (Precision.AlmostEquals(counter.Result.Value, 0, 1)) return false; float rotationSpeed = (float)(1.01 * spinner.HitObject.SpinsRequired / spinner.HitObject.Duration); return Precision.AlmostEquals(counter.Result.Value, rotationSpeed * 1000 * 60, 1); } }); } [Test] public void TestSpinnerGetsNoBonusScore() { DrawableSpinner? spinner = null; List results = new List(); CreateModTest(new ModTestData { Mod = new OsuModSpunOut(), Autoplay = false, Beatmap = singleSpinnerBeatmap, PassCondition = () => { // Bind to the first spinner's results for further tracking. if (spinner == null) { // We only care about the first spinner we encounter for this test. var nextSpinner = Player.ChildrenOfType().SingleOrDefault(); if (nextSpinner == null) return false; spinner = nextSpinner; spinner.OnNewResult += (_, result) => results.Add(result); results.Clear(); } // we should only be checking the bonus/progress after the spinner has fully completed. if (results.OfType().All(r => r.TimeCompleted == null)) return false; return results.Any(r => r.Type == HitResult.SmallBonus) && results.All(r => r.Type != HitResult.LargeBonus); } }); } private Beatmap singleSpinnerBeatmap => new Beatmap { HitObjects = new List { new Spinner { Position = new Vector2(256, 192), StartTime = 500, Duration = 2000 } } }; } }