2019-12-12 15:09:42 +08: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;
|
2022-11-30 16:35:50 +08:00
|
|
|
using System.Diagnostics;
|
2020-02-04 04:43:04 +08:00
|
|
|
using System.Linq;
|
2019-12-12 15:09:42 +08:00
|
|
|
using NUnit.Framework;
|
|
|
|
using osu.Framework.Allocation;
|
2022-10-30 20:24:58 +08:00
|
|
|
using osu.Framework.Extensions.ObjectExtensions;
|
2019-12-12 15:09:42 +08:00
|
|
|
using osu.Framework.Graphics;
|
2020-02-02 23:39:39 +08:00
|
|
|
using osu.Framework.Graphics.Containers;
|
2020-02-04 04:43:04 +08:00
|
|
|
using osu.Framework.Testing;
|
2022-03-18 04:22:07 +08:00
|
|
|
using osu.Framework.Timing;
|
2019-12-12 15:09:42 +08:00
|
|
|
using osu.Game.Configuration;
|
2022-10-30 20:38:31 +08:00
|
|
|
using osu.Game.Graphics.Containers;
|
2019-12-12 15:09:42 +08:00
|
|
|
using osu.Game.Rulesets.Mods;
|
2022-03-14 14:51:10 +08:00
|
|
|
using osu.Game.Rulesets.Osu;
|
2021-05-03 15:47:47 +08:00
|
|
|
using osu.Game.Rulesets.Scoring;
|
2019-12-12 15:09:42 +08:00
|
|
|
using osu.Game.Screens.Play;
|
2022-10-10 15:05:38 +08:00
|
|
|
using osu.Game.Screens.Play.HUD;
|
2022-06-14 17:35:33 +08:00
|
|
|
using osu.Game.Screens.Play.HUD.HitErrorMeters;
|
2021-08-16 11:56:59 +08:00
|
|
|
using osu.Game.Skinning;
|
2022-05-28 20:55:57 +08:00
|
|
|
using osu.Game.Tests.Gameplay;
|
2020-02-02 19:50:05 +08:00
|
|
|
using osuTK.Input;
|
2019-12-12 15:09:42 +08:00
|
|
|
|
|
|
|
namespace osu.Game.Tests.Visual.Gameplay
|
|
|
|
{
|
2020-10-30 13:20:00 +08:00
|
|
|
public partial class TestSceneHUDOverlay : OsuManualInputManagerTestScene
|
2019-12-12 15:09:42 +08:00
|
|
|
{
|
2022-10-30 20:24:58 +08:00
|
|
|
private OsuConfigManager localConfig = null!;
|
2021-08-18 09:59:08 +08:00
|
|
|
|
2022-10-30 20:24:58 +08:00
|
|
|
private HUDOverlay hudOverlay = null!;
|
2019-12-12 15:09:42 +08:00
|
|
|
|
2021-05-03 15:47:47 +08:00
|
|
|
[Cached]
|
2022-03-14 14:51:10 +08:00
|
|
|
private ScoreProcessor scoreProcessor = new ScoreProcessor(new OsuRuleset());
|
2021-05-03 15:47:47 +08:00
|
|
|
|
2021-05-07 16:31:29 +08:00
|
|
|
[Cached(typeof(HealthProcessor))]
|
2021-05-07 16:47:33 +08:00
|
|
|
private HealthProcessor healthProcessor = new DrainingHealthProcessor(0);
|
2021-05-07 15:56:24 +08:00
|
|
|
|
2022-03-15 21:55:47 +08:00
|
|
|
[Cached]
|
2022-05-28 20:55:57 +08:00
|
|
|
private GameplayState gameplayState = TestGameplayState.Create(new OsuRuleset());
|
2022-03-15 21:55:47 +08:00
|
|
|
|
2022-08-15 16:36:18 +08:00
|
|
|
[Cached(typeof(IGameplayClock))]
|
2022-08-15 18:46:29 +08:00
|
|
|
private readonly IGameplayClock gameplayClock = new GameplayClockContainer(new FramedClock());
|
2022-03-18 04:22:07 +08:00
|
|
|
|
2020-02-04 04:17:10 +08:00
|
|
|
// best way to check without exposing.
|
|
|
|
private Drawable hideTarget => hudOverlay.KeyCounter;
|
2020-02-04 04:43:04 +08:00
|
|
|
private FillFlowContainer<KeyCounter> keyCounterFlow => hudOverlay.KeyCounter.ChildrenOfType<FillFlowContainer<KeyCounter>>().First();
|
2019-12-12 15:09:42 +08:00
|
|
|
|
2021-08-18 09:59:08 +08:00
|
|
|
[BackgroundDependencyLoader]
|
|
|
|
private void load()
|
|
|
|
{
|
|
|
|
Dependencies.Cache(localConfig = new OsuConfigManager(LocalStorage));
|
|
|
|
}
|
|
|
|
|
2021-08-18 11:50:01 +08:00
|
|
|
[SetUp]
|
2021-08-18 09:59:08 +08:00
|
|
|
public void SetUp() => Schedule(() => localConfig.SetValue(OsuSetting.HUDVisibilityMode, HUDVisibilityMode.Always));
|
2019-12-12 15:09:42 +08:00
|
|
|
|
2020-10-14 16:20:44 +08:00
|
|
|
[Test]
|
|
|
|
public void TestComboCounterIncrementing()
|
|
|
|
{
|
|
|
|
createNew();
|
|
|
|
|
2021-05-03 15:47:47 +08:00
|
|
|
AddRepeatStep("increase combo", () => { scoreProcessor.Combo.Value++; }, 10);
|
2020-10-14 16:20:44 +08:00
|
|
|
|
2021-05-03 15:47:47 +08:00
|
|
|
AddStep("reset combo", () => { scoreProcessor.Combo.Value = 0; });
|
2020-10-14 16:20:44 +08:00
|
|
|
}
|
|
|
|
|
2019-12-12 15:09:42 +08:00
|
|
|
[Test]
|
|
|
|
public void TestShownByDefault()
|
|
|
|
{
|
|
|
|
createNew();
|
|
|
|
|
|
|
|
AddAssert("showhud is set", () => hudOverlay.ShowHud.Value);
|
|
|
|
|
|
|
|
AddAssert("hidetarget is visible", () => hideTarget.IsPresent);
|
2020-02-04 04:17:10 +08:00
|
|
|
AddAssert("key counter flow is visible", () => keyCounterFlow.IsPresent);
|
2019-12-12 15:09:42 +08:00
|
|
|
AddAssert("pause button is visible", () => hudOverlay.HoldToQuit.IsPresent);
|
|
|
|
}
|
|
|
|
|
|
|
|
[Test]
|
|
|
|
public void TestFadesInOnLoadComplete()
|
|
|
|
{
|
|
|
|
float? initialAlpha = null;
|
|
|
|
|
|
|
|
createNew(h => h.OnLoadComplete += _ => initialAlpha = hideTarget.Alpha);
|
|
|
|
AddUntilStep("wait for load", () => hudOverlay.IsAlive);
|
2020-10-16 11:49:39 +08:00
|
|
|
AddAssert("initial alpha was less than 1", () => initialAlpha < 1);
|
2019-12-12 15:09:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
[Test]
|
2020-02-04 00:59:58 +08:00
|
|
|
public void TestHideExternally()
|
2019-12-12 15:09:42 +08:00
|
|
|
{
|
|
|
|
createNew();
|
|
|
|
|
2020-10-30 13:20:00 +08:00
|
|
|
AddStep("set showhud false", () => hudOverlay.ShowHud.Value = false);
|
2020-02-04 00:59:58 +08:00
|
|
|
|
2019-12-12 15:09:42 +08:00
|
|
|
AddUntilStep("hidetarget is hidden", () => !hideTarget.IsPresent);
|
|
|
|
AddAssert("pause button is still visible", () => hudOverlay.HoldToQuit.IsPresent);
|
2020-02-02 19:50:05 +08:00
|
|
|
|
2020-02-04 00:59:58 +08:00
|
|
|
// Key counter flow container should not be affected by this, only the key counter display will be hidden as checked above.
|
2020-02-04 04:17:10 +08:00
|
|
|
AddAssert("key counter flow not affected", () => keyCounterFlow.IsPresent);
|
2019-12-12 15:09:42 +08:00
|
|
|
}
|
|
|
|
|
2020-10-30 13:20:00 +08:00
|
|
|
[Test]
|
|
|
|
public void TestMomentaryShowHUD()
|
|
|
|
{
|
|
|
|
createNew();
|
|
|
|
|
2021-08-18 09:59:08 +08:00
|
|
|
AddStep("set hud to never show", () => localConfig.SetValue(OsuSetting.HUDVisibilityMode, HUDVisibilityMode.Never));
|
2020-10-30 13:20:00 +08:00
|
|
|
|
|
|
|
AddUntilStep("wait for fade", () => !hideTarget.IsPresent);
|
|
|
|
|
|
|
|
AddStep("trigger momentary show", () => InputManager.PressKey(Key.ControlLeft));
|
|
|
|
AddUntilStep("wait for visible", () => hideTarget.IsPresent);
|
|
|
|
|
|
|
|
AddStep("stop trigering", () => InputManager.ReleaseKey(Key.ControlLeft));
|
|
|
|
AddUntilStep("wait for fade", () => !hideTarget.IsPresent);
|
|
|
|
}
|
|
|
|
|
2019-12-12 15:09:42 +08:00
|
|
|
[Test]
|
|
|
|
public void TestExternalHideDoesntAffectConfig()
|
|
|
|
{
|
|
|
|
createNew();
|
|
|
|
|
|
|
|
AddStep("set showhud false", () => hudOverlay.ShowHud.Value = false);
|
2021-08-18 09:59:08 +08:00
|
|
|
AddAssert("config unchanged", () => localConfig.GetBindable<HUDVisibilityMode>(OsuSetting.HUDVisibilityMode).IsDefault);
|
2019-12-12 15:09:42 +08:00
|
|
|
|
|
|
|
AddStep("set showhud true", () => hudOverlay.ShowHud.Value = true);
|
2021-08-18 09:59:08 +08:00
|
|
|
AddAssert("config unchanged", () => localConfig.GetBindable<HUDVisibilityMode>(OsuSetting.HUDVisibilityMode).IsDefault);
|
2019-12-12 15:09:42 +08:00
|
|
|
}
|
|
|
|
|
2020-02-02 19:50:05 +08:00
|
|
|
[Test]
|
|
|
|
public void TestChangeHUDVisibilityOnHiddenKeyCounter()
|
|
|
|
{
|
|
|
|
createNew();
|
|
|
|
|
2021-08-18 09:59:08 +08:00
|
|
|
AddStep("hide key overlay", () =>
|
2020-02-04 00:59:58 +08:00
|
|
|
{
|
2021-08-18 09:59:08 +08:00
|
|
|
localConfig.SetValue(OsuSetting.KeyOverlay, false);
|
2020-10-30 13:20:00 +08:00
|
|
|
hudOverlay.KeyCounter.AlwaysVisible.Value = false;
|
2020-02-04 00:59:58 +08:00
|
|
|
});
|
2020-02-02 19:50:05 +08:00
|
|
|
|
2020-10-30 13:20:00 +08:00
|
|
|
AddStep("set showhud false", () => hudOverlay.ShowHud.Value = false);
|
2020-02-02 19:50:05 +08:00
|
|
|
AddUntilStep("hidetarget is hidden", () => !hideTarget.IsPresent);
|
2020-02-04 04:17:10 +08:00
|
|
|
AddAssert("key counters hidden", () => !keyCounterFlow.IsPresent);
|
2020-02-02 19:50:05 +08:00
|
|
|
|
2020-10-30 13:20:00 +08:00
|
|
|
AddStep("set showhud true", () => hudOverlay.ShowHud.Value = true);
|
2020-02-02 19:50:05 +08:00
|
|
|
AddUntilStep("hidetarget is visible", () => hideTarget.IsPresent);
|
2020-02-04 04:17:10 +08:00
|
|
|
AddAssert("key counters still hidden", () => !keyCounterFlow.IsPresent);
|
2020-02-02 19:50:05 +08:00
|
|
|
}
|
|
|
|
|
2022-10-30 20:38:31 +08:00
|
|
|
[Test]
|
|
|
|
public void TestHoldForMenuDoesWorkWhenHidden()
|
|
|
|
{
|
|
|
|
bool activated = false;
|
|
|
|
|
|
|
|
HoldForMenuButton getHoldForMenu() => hudOverlay.ChildrenOfType<HoldForMenuButton>().Single();
|
|
|
|
|
|
|
|
createNew();
|
|
|
|
|
|
|
|
AddStep("bind action", () =>
|
|
|
|
{
|
|
|
|
activated = false;
|
|
|
|
|
|
|
|
var holdForMenu = getHoldForMenu();
|
|
|
|
|
|
|
|
holdForMenu.Action += () => activated = true;
|
|
|
|
});
|
|
|
|
|
|
|
|
AddStep("set showhud false", () => hudOverlay.ShowHud.Value = false);
|
|
|
|
AddUntilStep("hidetarget is hidden", () => !hideTarget.IsPresent);
|
|
|
|
|
|
|
|
AddStep("attempt activate", () =>
|
|
|
|
{
|
|
|
|
InputManager.MoveMouseTo(getHoldForMenu().OfType<HoldToConfirmContainer>().Single());
|
|
|
|
InputManager.PressButton(MouseButton.Left);
|
|
|
|
});
|
|
|
|
|
|
|
|
AddUntilStep("activated", () => activated);
|
|
|
|
|
|
|
|
AddStep("release mouse button", () =>
|
|
|
|
{
|
|
|
|
InputManager.ReleaseButton(MouseButton.Left);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-10-10 15:05:38 +08:00
|
|
|
[Test]
|
|
|
|
public void TestInputDoesntWorkWhenHUDHidden()
|
|
|
|
{
|
2023-01-18 14:10:01 +08:00
|
|
|
ArgonSongProgress? getSongProgress() => hudOverlay.ChildrenOfType<ArgonSongProgress>().SingleOrDefault();
|
2022-10-10 15:05:38 +08:00
|
|
|
|
|
|
|
bool seeked = false;
|
|
|
|
|
|
|
|
createNew();
|
|
|
|
|
2022-11-30 16:35:50 +08:00
|
|
|
AddUntilStep("wait for song progress", () => getSongProgress() != null);
|
|
|
|
|
2022-10-10 15:05:38 +08:00
|
|
|
AddStep("bind seek", () =>
|
|
|
|
{
|
|
|
|
seeked = false;
|
|
|
|
|
|
|
|
var progress = getSongProgress();
|
|
|
|
|
2022-11-30 16:35:50 +08:00
|
|
|
Debug.Assert(progress != null);
|
|
|
|
|
2023-01-18 14:10:01 +08:00
|
|
|
progress.Interactive.Value = true;
|
|
|
|
progress.ChildrenOfType<ArgonSongProgressBar>().Single().OnSeek += _ => seeked = true;
|
2022-10-10 15:05:38 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
AddStep("set showhud false", () => hudOverlay.ShowHud.Value = false);
|
|
|
|
AddUntilStep("hidetarget is hidden", () => !hideTarget.IsPresent);
|
|
|
|
|
|
|
|
AddStep("attempt seek", () =>
|
|
|
|
{
|
2023-01-18 13:37:09 +08:00
|
|
|
InputManager.MoveMouseTo(getSongProgress() as Drawable);
|
2022-10-10 15:05:38 +08:00
|
|
|
InputManager.Click(MouseButton.Left);
|
|
|
|
});
|
|
|
|
|
|
|
|
AddAssert("seek not performed", () => !seeked);
|
|
|
|
|
|
|
|
AddStep("set showhud true", () => hudOverlay.ShowHud.Value = true);
|
|
|
|
|
|
|
|
AddStep("attempt seek", () => InputManager.Click(MouseButton.Left));
|
|
|
|
AddAssert("seek performed", () => seeked);
|
|
|
|
}
|
|
|
|
|
2022-06-14 17:35:33 +08:00
|
|
|
[Test]
|
|
|
|
public void TestHiddenHUDDoesntBlockComponentUpdates()
|
|
|
|
{
|
|
|
|
int updateCount = 0;
|
|
|
|
|
|
|
|
AddStep("set hud to never show", () => localConfig.SetValue(OsuSetting.HUDVisibilityMode, HUDVisibilityMode.Never));
|
|
|
|
|
|
|
|
createNew();
|
|
|
|
|
|
|
|
AddUntilStep("wait for hud load", () => hudOverlay.IsLoaded);
|
|
|
|
AddUntilStep("wait for components to be hidden", () => hudOverlay.ChildrenOfType<SkinnableTargetContainer>().Single().Alpha == 0);
|
2022-07-28 16:58:07 +08:00
|
|
|
AddUntilStep("wait for hud load", () => hudOverlay.ChildrenOfType<SkinnableTargetContainer>().All(c => c.ComponentsLoaded));
|
2022-06-14 17:35:33 +08:00
|
|
|
|
|
|
|
AddStep("bind on update", () =>
|
|
|
|
{
|
|
|
|
hudOverlay.ChildrenOfType<BarHitErrorMeter>().First().OnUpdate += _ => updateCount++;
|
|
|
|
});
|
|
|
|
|
|
|
|
AddUntilStep("wait for updates", () => updateCount > 0);
|
|
|
|
}
|
|
|
|
|
2021-08-16 11:56:59 +08:00
|
|
|
[Test]
|
|
|
|
public void TestHiddenHUDDoesntBlockSkinnableComponentsLoad()
|
|
|
|
{
|
2021-08-18 09:59:08 +08:00
|
|
|
AddStep("set hud to never show", () => localConfig.SetValue(OsuSetting.HUDVisibilityMode, HUDVisibilityMode.Never));
|
2021-08-16 11:56:59 +08:00
|
|
|
|
|
|
|
createNew();
|
2021-08-24 14:18:05 +08:00
|
|
|
|
2021-08-16 11:56:59 +08:00
|
|
|
AddUntilStep("wait for hud load", () => hudOverlay.IsLoaded);
|
2022-06-14 17:35:33 +08:00
|
|
|
AddUntilStep("wait for components to be hidden", () => hudOverlay.ChildrenOfType<SkinnableTargetContainer>().Single().Alpha == 0);
|
2021-08-24 14:18:05 +08:00
|
|
|
|
|
|
|
AddStep("reload components", () => hudOverlay.ChildrenOfType<SkinnableTargetContainer>().Single().Reload());
|
2021-08-16 11:56:59 +08:00
|
|
|
AddUntilStep("skinnable components loaded", () => hudOverlay.ChildrenOfType<SkinnableTargetContainer>().Single().ComponentsLoaded);
|
|
|
|
}
|
|
|
|
|
2022-10-30 20:24:58 +08:00
|
|
|
private void createNew(Action<HUDOverlay>? action = null)
|
2019-12-12 15:09:42 +08:00
|
|
|
{
|
|
|
|
AddStep("create overlay", () =>
|
|
|
|
{
|
2021-05-07 16:27:34 +08:00
|
|
|
hudOverlay = new HUDOverlay(null, Array.Empty<Mod>());
|
2019-12-12 15:09:42 +08:00
|
|
|
|
2020-10-30 13:20:00 +08:00
|
|
|
// Add any key just to display the key counter visually.
|
|
|
|
hudOverlay.KeyCounter.Add(new KeyCounterKeyboard(Key.Space));
|
2019-12-12 15:09:42 +08:00
|
|
|
|
2021-05-03 15:47:47 +08:00
|
|
|
scoreProcessor.Combo.Value = 1;
|
2020-02-02 19:50:05 +08:00
|
|
|
|
2020-10-30 13:20:00 +08:00
|
|
|
action?.Invoke(hudOverlay);
|
2020-10-14 15:45:48 +08:00
|
|
|
|
2020-10-30 13:20:00 +08:00
|
|
|
Child = hudOverlay;
|
2020-02-04 04:17:10 +08:00
|
|
|
});
|
2020-02-02 19:50:05 +08:00
|
|
|
}
|
2021-08-18 09:59:08 +08:00
|
|
|
|
|
|
|
protected override void Dispose(bool isDisposing)
|
|
|
|
{
|
2022-10-30 20:24:58 +08:00
|
|
|
if (localConfig.IsNotNull())
|
|
|
|
localConfig.Dispose();
|
|
|
|
|
2021-08-18 09:59:08 +08:00
|
|
|
base.Dispose(isDisposing);
|
|
|
|
}
|
2019-12-12 15:09:42 +08:00
|
|
|
}
|
|
|
|
}
|