// 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.Linq; using NUnit.Framework; using osu.Framework.Allocation; using osu.Framework.Graphics; using osu.Framework.Graphics.Containers; using osu.Framework.Logging; using osu.Game.Graphics.UserInterface; using osu.Game.Input.Bindings; using osu.Game.Screens.Play; using osuTK; using osuTK.Input; namespace osu.Game.Tests.Visual.Gameplay { [Description("player pause/fail screens")] public class TestSceneGameplayMenuOverlay : OsuManualInputManagerTestScene { private FailOverlay failOverlay; private PauseOverlay pauseOverlay; private GlobalActionContainer globalActionContainer; [BackgroundDependencyLoader] private void load(OsuGameBase game) { Child = globalActionContainer = new GlobalActionContainer(game); } [SetUp] public void SetUp() => Schedule(() => { globalActionContainer.Children = new Drawable[] { pauseOverlay = new PauseOverlay { OnResume = () => Logger.Log(@"Resume"), OnRetry = () => Logger.Log(@"Retry"), OnQuit = () => Logger.Log(@"Quit"), }, failOverlay = new FailOverlay { OnRetry = () => Logger.Log(@"Retry"), OnQuit = () => Logger.Log(@"Quit"), } }; InputManager.MoveMouseTo(Vector2.Zero); }); [Test] public void TestAdjustRetryCount() { showOverlay(); var retryCount = 0; AddRepeatStep("Add retry", () => { retryCount++; pauseOverlay.Retries = failOverlay.Retries = retryCount; }, 10); } /// /// Tests that pressing enter after an overlay shows doesn't trigger an event because a selection hasn't occurred. /// [Test] public void TestEnterWithoutSelection() { showOverlay(); AddStep("Press select", () => press(GlobalAction.Select)); AddAssert("Overlay still open", () => pauseOverlay.State.Value == Visibility.Visible); } /// /// Tests that pressing the up arrow from the initial state selects the last button. /// [Test] public void TestKeyUpFromInitial() { showOverlay(); AddStep("Up arrow", () => InputManager.Key(Key.Up)); AddAssert("Last button selected", () => pauseOverlay.Buttons.Last().State == SelectionState.Selected); } /// /// Tests that pressing the down arrow from the initial state selects the first button. /// [Test] public void TestKeyDownFromInitial() { showOverlay(); AddStep("Down arrow", () => InputManager.Key(Key.Down)); AddAssert("First button selected", () => getButton(0).State == SelectionState.Selected); } /// /// Tests that pressing the up arrow repeatedly causes the selected button to wrap correctly. /// [Test] public void TestKeyUpWrapping() { AddStep("Show overlay", () => failOverlay.Show()); AddStep("Up arrow", () => InputManager.Key(Key.Up)); AddAssert("Last button selected", () => failOverlay.Buttons.Last().State == SelectionState.Selected); AddStep("Up arrow", () => InputManager.Key(Key.Up)); AddAssert("First button selected", () => failOverlay.Buttons.First().State == SelectionState.Selected); AddStep("Up arrow", () => InputManager.Key(Key.Up)); AddAssert("Last button selected", () => failOverlay.Buttons.Last().State == SelectionState.Selected); } /// /// Tests that pressing the down arrow repeatedly causes the selected button to wrap correctly. /// [Test] public void TestKeyDownWrapping() { AddStep("Show overlay", () => failOverlay.Show()); AddStep("Down arrow", () => InputManager.Key(Key.Down)); AddAssert("First button selected", () => failOverlay.Buttons.First().State == SelectionState.Selected); AddStep("Down arrow", () => InputManager.Key(Key.Down)); AddAssert("Last button selected", () => failOverlay.Buttons.Last().State == SelectionState.Selected); AddStep("Down arrow", () => InputManager.Key(Key.Down)); AddAssert("First button selected", () => failOverlay.Buttons.First().State == SelectionState.Selected); } /// /// Test that hiding the overlay after hovering a button will reset the overlay to the initial state with no buttons selected. /// [Test] public void TestHideResets() { AddStep("Show overlay", () => failOverlay.Show()); AddStep("Hover first button", () => InputManager.MoveMouseTo(failOverlay.Buttons.First())); AddStep("Hide overlay", () => failOverlay.Hide()); AddAssert("Overlay state is reset", () => failOverlay.Buttons.All(b => b.State == SelectionState.NotSelected)); } /// /// Tests that entering menu with cursor initially on button doesn't selects it immediately. /// This is to allow for stable keyboard navigation. /// [Test] public void TestInitialButtonHover() { showOverlay(); AddStep("Hover first button", () => InputManager.MoveMouseTo(getButton(0))); AddStep("Hide overlay", () => pauseOverlay.Hide()); showOverlay(); AddAssert("First button not selected", () => getButton(0).State == SelectionState.NotSelected); AddStep("Move slightly", () => InputManager.MoveMouseTo(InputManager.CurrentState.Mouse.Position + new Vector2(1))); AddAssert("First button selected", () => getButton(0).State == SelectionState.Selected); } /// /// Tests that hovering a button that was previously selected with the keyboard correctly selects the new button and deselects the previous button. /// [Test] public void TestMouseSelectionAfterKeySelection() { showOverlay(); AddStep("Down arrow", () => InputManager.Key(Key.Down)); AddStep("Hover second button", () => InputManager.MoveMouseTo(getButton(1))); AddAssert("First button not selected", () => getButton(0).State == SelectionState.NotSelected); AddAssert("Second button selected", () => getButton(1).State == SelectionState.Selected); } /// /// Tests that pressing a key after selecting a button with a hover event correctly selects a new button and deselects the previous button. /// [Test] public void TestKeySelectionAfterMouseSelection() { AddStep("Show overlay", () => { pauseOverlay.Show(); }); AddStep("Hover second button", () => InputManager.MoveMouseTo(getButton(1))); AddStep("Up arrow", () => InputManager.Key(Key.Up)); AddAssert("Second button not selected", () => getButton(1).State == SelectionState.NotSelected); AddAssert("First button selected", () => getButton(0).State == SelectionState.Selected); } /// /// Tests that deselecting with the mouse by losing hover will reset the overlay to the initial state. /// [Test] public void TestMouseDeselectionResets() { showOverlay(); AddStep("Hover second button", () => InputManager.MoveMouseTo(getButton(1))); AddStep("Unhover second button", () => InputManager.MoveMouseTo(Vector2.Zero)); AddStep("Down arrow", () => InputManager.Key(Key.Down)); AddAssert("First button selected", () => getButton(0).State == SelectionState.Selected); // Initial state condition } /// /// Tests that clicking on a button correctly causes a click event for that button. /// [Test] public void TestClickSelection() { showOverlay(); bool triggered = false; AddStep("Click retry button", () => { var lastAction = pauseOverlay.OnRetry; pauseOverlay.OnRetry = () => triggered = true; getButton(1).Click(); pauseOverlay.OnRetry = lastAction; }); AddAssert("Action was triggered", () => triggered); AddAssert("Overlay is closed", () => pauseOverlay.State.Value == Visibility.Hidden); } /// /// Tests that pressing the enter key with a button selected correctly causes a click event for that button. /// [Test] public void TestEnterKeySelection() { showOverlay(); AddStep("Select second button", () => { InputManager.Key(Key.Down); InputManager.Key(Key.Down); }); bool triggered = false; Action lastAction = null; AddStep("Press enter", () => { lastAction = pauseOverlay.OnRetry; pauseOverlay.OnRetry = () => triggered = true; InputManager.Key(Key.Enter); }); AddAssert("Action was triggered", () => { if (lastAction != null) { pauseOverlay.OnRetry = lastAction; lastAction = null; } return triggered; }); AddAssert("Overlay is closed", () => pauseOverlay.State.Value == Visibility.Hidden); } [Test] public void TestSelectionResetOnVisibilityChange() { showOverlay(); AddStep("Select last button", () => InputManager.Key(Key.Up)); hideOverlay(); showOverlay(); AddAssert("No button selected", () => pauseOverlay.Buttons.All(button => button.State == SelectionState.NotSelected)); } private void showOverlay() => AddStep("Show overlay", () => pauseOverlay.Show()); private void hideOverlay() => AddStep("Hide overlay", () => pauseOverlay.Hide()); private DialogButton getButton(int index) => pauseOverlay.Buttons.Skip(index).First(); private void press(GlobalAction action) { globalActionContainer.TriggerPressed(action); globalActionContainer.TriggerReleased(action); } } }