// Copyright (c) ppy Pty Ltd . Licensed under the MIT Licence. // See the LICENCE file in the repository root for full licence text. using System.Collections.Generic; using System.Linq; using NUnit.Framework; using osu.Framework.Screens; using osu.Game.Beatmaps; using osu.Game.Beatmaps.ControlPoints; using osu.Game.Replays; using osu.Game.Rulesets.Judgements; using osu.Game.Rulesets.Mania.Objects; using osu.Game.Rulesets.Mania.Replays; using osu.Game.Rulesets.Mania.Scoring; using osu.Game.Rulesets.Objects; using osu.Game.Rulesets.Replays; using osu.Game.Rulesets.Scoring; using osu.Game.Scoring; using osu.Game.Screens.Play; using osu.Game.Tests.Visual; namespace osu.Game.Rulesets.Mania.Tests { public class TestSceneHoldNoteInput : RateAdjustedBeatmapTestScene { private const double time_before_head = 250; private const double time_head = 1500; private const double time_during_hold_1 = 2500; private const double time_tail = 4000; private const double time_after_tail = 5250; private List judgementResults; /// /// -----[ ]----- /// o o /// [Test] public void TestNoInput() { performTest(new List { new ManiaReplayFrame(time_before_head), new ManiaReplayFrame(time_after_tail), }); assertHeadJudgement(HitResult.Miss); assertTickJudgement(HitResult.Miss); assertTailJudgement(HitResult.Miss); assertNoteJudgement(HitResult.Perfect); } /// /// -----[ ]----- /// x o /// [Test] public void TestPressTooEarlyAndReleaseAfterTail() { performTest(new List { new ManiaReplayFrame(time_before_head, ManiaAction.Key1), new ManiaReplayFrame(time_after_tail, ManiaAction.Key1), }); assertHeadJudgement(HitResult.Miss); assertTickJudgement(HitResult.Miss); assertTailJudgement(HitResult.Miss); } /// /// -----[ ]----- /// x o /// [Test] public void TestPressTooEarlyAndReleaseAtTail() { performTest(new List { new ManiaReplayFrame(time_before_head, ManiaAction.Key1), new ManiaReplayFrame(time_tail), }); assertHeadJudgement(HitResult.Miss); assertTickJudgement(HitResult.Miss); assertTailJudgement(HitResult.Miss); } /// /// -----[ ]----- /// xo x o /// [Test] public void TestPressTooEarlyThenPressAtStartAndReleaseAfterTail() { performTest(new List { new ManiaReplayFrame(time_before_head, ManiaAction.Key1), new ManiaReplayFrame(time_before_head + 10), new ManiaReplayFrame(time_head, ManiaAction.Key1), new ManiaReplayFrame(time_after_tail), }); assertHeadJudgement(HitResult.Perfect); assertTickJudgement(HitResult.Perfect); assertTailJudgement(HitResult.Miss); } /// /// -----[ ]----- /// xo x o /// [Test] public void TestPressTooEarlyThenPressAtStartAndReleaseAtTail() { performTest(new List { new ManiaReplayFrame(time_before_head, ManiaAction.Key1), new ManiaReplayFrame(time_before_head + 10), new ManiaReplayFrame(time_head, ManiaAction.Key1), new ManiaReplayFrame(time_tail), }); assertHeadJudgement(HitResult.Perfect); assertTickJudgement(HitResult.Perfect); assertTailJudgement(HitResult.Perfect); } /// /// -----[ ]----- /// xo o /// [Test] public void TestPressAtStartAndBreak() { performTest(new List { new ManiaReplayFrame(time_head, ManiaAction.Key1), new ManiaReplayFrame(time_head + 10), new ManiaReplayFrame(time_after_tail), }); assertHeadJudgement(HitResult.Perfect); assertTickJudgement(HitResult.Miss); assertTailJudgement(HitResult.Miss); } /// /// -----[ ]----- /// xo x o /// [Test] public void TestPressAtStartThenBreakThenRepressAndReleaseAfterTail() { performTest(new List { new ManiaReplayFrame(time_head, ManiaAction.Key1), new ManiaReplayFrame(time_head + 10), new ManiaReplayFrame(time_during_hold_1, ManiaAction.Key1), new ManiaReplayFrame(time_after_tail), }); assertHeadJudgement(HitResult.Perfect); assertTickJudgement(HitResult.Perfect); assertTailJudgement(HitResult.Miss); } /// /// -----[ ]----- /// xo x o o /// [Test] public void TestPressAtStartThenBreakThenRepressAndReleaseAtTail() { performTest(new List { new ManiaReplayFrame(time_head, ManiaAction.Key1), new ManiaReplayFrame(time_head + 10), new ManiaReplayFrame(time_during_hold_1, ManiaAction.Key1), new ManiaReplayFrame(time_tail), }); assertHeadJudgement(HitResult.Perfect); assertTickJudgement(HitResult.Perfect); assertTailJudgement(HitResult.Meh); } /// /// -----[ ]----- /// x o /// [Test] public void TestPressDuringNoteAndReleaseAfterTail() { performTest(new List { new ManiaReplayFrame(time_during_hold_1, ManiaAction.Key1), new ManiaReplayFrame(time_after_tail), }); assertHeadJudgement(HitResult.Miss); assertTickJudgement(HitResult.Perfect); assertTailJudgement(HitResult.Miss); } /// /// -----[ ]----- /// x o o /// [Test] public void TestPressDuringNoteAndReleaseAtTail() { performTest(new List { new ManiaReplayFrame(time_during_hold_1, ManiaAction.Key1), new ManiaReplayFrame(time_tail), }); assertHeadJudgement(HitResult.Miss); assertTickJudgement(HitResult.Perfect); assertTailJudgement(HitResult.Meh); } /// /// -----[ ]----- /// xo o /// [Test] public void TestPressAndReleaseAtTail() { performTest(new List { new ManiaReplayFrame(time_tail, ManiaAction.Key1), new ManiaReplayFrame(time_tail + 10), }); assertHeadJudgement(HitResult.Miss); assertTickJudgement(HitResult.Miss); assertTailJudgement(HitResult.Meh); } [Test] public void TestMissReleaseAndHitSecondRelease() { var windows = new ManiaHitWindows(); windows.SetDifficulty(10); var beatmap = new Beatmap { HitObjects = { new HoldNote { StartTime = 1000, Duration = 500, Column = 0, }, new HoldNote { StartTime = 1000 + 500 + windows.WindowFor(HitResult.Miss) + 10, Duration = 500, Column = 0, }, }, BeatmapInfo = { BaseDifficulty = new BeatmapDifficulty { SliderTickRate = 4, OverallDifficulty = 10, }, Ruleset = new ManiaRuleset().RulesetInfo }, }; performTest(new List { new ManiaReplayFrame(beatmap.HitObjects[1].StartTime, ManiaAction.Key1), new ManiaReplayFrame(beatmap.HitObjects[1].GetEndTime()), }, beatmap); AddAssert("first hold note missed", () => judgementResults.Where(j => beatmap.HitObjects[0].NestedHitObjects.Contains(j.HitObject)) .All(j => j.Type == HitResult.Miss)); AddAssert("second hold note missed", () => judgementResults.Where(j => beatmap.HitObjects[1].NestedHitObjects.Contains(j.HitObject)) .All(j => j.Type == HitResult.Perfect)); } private void assertHeadJudgement(HitResult result) => AddAssert($"head judged as {result}", () => judgementResults[0].Type == result); private void assertTailJudgement(HitResult result) => AddAssert($"tail judged as {result}", () => judgementResults[^2].Type == result); private void assertNoteJudgement(HitResult result) => AddAssert($"hold note judged as {result}", () => judgementResults[^1].Type == result); private void assertTickJudgement(HitResult result) => AddAssert($"tick judged as {result}", () => judgementResults[6].Type == result); // arbitrary tick private ScoreAccessibleReplayPlayer currentPlayer; private void performTest(List frames, Beatmap beatmap = null) { if (beatmap == null) { beatmap = new Beatmap { HitObjects = { new HoldNote { StartTime = time_head, Duration = time_tail - time_head, Column = 0, } }, BeatmapInfo = { BaseDifficulty = new BeatmapDifficulty { SliderTickRate = 4 }, Ruleset = new ManiaRuleset().RulesetInfo }, }; beatmap.ControlPointInfo.Add(0, new DifficultyControlPoint { SpeedMultiplier = 0.1f }); } AddStep("load player", () => { Beatmap.Value = CreateWorkingBeatmap(beatmap); var p = new ScoreAccessibleReplayPlayer(new Score { Replay = new Replay { Frames = frames } }); p.OnLoadComplete += _ => { p.ScoreProcessor.NewJudgement += result => { if (currentPlayer == p) judgementResults.Add(result); }; }; LoadScreen(currentPlayer = p); judgementResults = new List(); }); AddUntilStep("Beatmap at 0", () => MusicController.CurrentTrack?.CurrentTime == 0); AddUntilStep("Wait until player is loaded", () => currentPlayer.IsCurrentScreen()); AddUntilStep("Wait for completion", () => currentPlayer.ScoreProcessor.HasCompleted.Value); } private class ScoreAccessibleReplayPlayer : ReplayPlayer { public new ScoreProcessor ScoreProcessor => base.ScoreProcessor; protected override bool PauseOnFocusLost => false; public ScoreAccessibleReplayPlayer(Score score) : base(score, false, false) { } } } }