// Copyright (c) ppy Pty Ltd . Licensed under the MIT Licence. // See the LICENCE file in the repository root for full licence text. #nullable disable using System; using System.Diagnostics; using System.Linq; using JetBrains.Annotations; using osu.Framework.Allocation; using osu.Framework.Development; using osu.Framework.Extensions.IEnumerableExtensions; using osu.Framework.Testing; using osu.Game.Beatmaps; using osu.Game.Configuration; using osu.Game.Database; using osu.Game.Rulesets; using osu.Game.Rulesets.Mods; namespace osu.Game.Tests.Visual { public abstract class PlayerTestScene : RateAdjustedBeatmapTestScene { /// /// Whether custom test steps are provided. Custom tests should invoke to create the test steps. /// protected virtual bool HasCustomSteps => false; /// /// Import the beatmap to the database before starting gameplay. Handy for testing local score storage and the likes. /// /// /// Only works under headless operation currently due to realm isolation difficulties. /// If this is ever needed to change, consideration needs to be given to the fact that BeatmapManager is attached to the global realm. /// protected virtual bool ImportBeatmapToDatabase => false; protected TestPlayer Player; protected OsuConfigManager LocalConfig; [BackgroundDependencyLoader] private void load() { Dependencies.Cache(LocalConfig = new OsuConfigManager(LocalStorage)); LocalConfig.GetBindable(OsuSetting.DimLevel).Value = 1.0; } [SetUpSteps] public override void SetUpSteps() { base.SetUpSteps(); if (!HasCustomSteps) CreateTest(); } protected void CreateTest([CanBeNull] Action action = null) { if (action != null && !HasCustomSteps) throw new InvalidOperationException($"Cannot add custom test steps without {nameof(HasCustomSteps)} being set."); action?.Invoke(); AddStep($"Load player for {CreatePlayerRuleset().Description}", LoadPlayer); AddUntilStep("player loaded", () => Player.IsLoaded && Player.Alpha == 1); } protected virtual bool AllowFail => false; protected virtual bool Autoplay => false; [Resolved] private BeatmapManager beatmaps { get; set; } protected void LoadPlayer() { var ruleset = CreatePlayerRuleset(); Ruleset.Value = ruleset.RulesetInfo; var beatmap = CreateBeatmap(ruleset.RulesetInfo); if (ImportBeatmapToDatabase) { Debug.Assert(DebugUtils.IsNUnitRunning, $@"Importing beatmaps in {nameof(PlayerTestScene)} requires headless environment."); Debug.Assert(beatmap.BeatmapInfo.BeatmapSet != null); var imported = beatmaps.Import(beatmap.BeatmapInfo.BeatmapSet); Debug.Assert(imported != null); beatmap.BeatmapInfo = null; beatmap.Difficulty = null; beatmap.BeatmapInfo = imported.Value.Detach().Beatmaps.First(); } Beatmap.Value = CreateWorkingBeatmap(beatmap); SelectedMods.Value = Array.Empty(); if (!AllowFail) { var noFailMod = ruleset.CreateMod(); if (noFailMod != null) SelectedMods.Value = new[] { noFailMod }; } if (Autoplay) { var mod = ruleset.GetAutoplayMod(); if (mod != null) SelectedMods.Value = SelectedMods.Value.Concat(mod.Yield()).ToArray(); } Player = CreatePlayer(ruleset); LoadScreen(Player); } protected override void Dispose(bool isDisposing) { LocalConfig?.Dispose(); base.Dispose(isDisposing); } /// /// Creates the ruleset for setting up the component. /// [NotNull] protected abstract Ruleset CreatePlayerRuleset(); protected sealed override Ruleset CreateRuleset() => CreatePlayerRuleset(); protected virtual TestPlayer CreatePlayer(Ruleset ruleset) => new TestPlayer(false, false); } }