// 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 osu.Framework.Allocation; using osu.Framework.Bindables; using osu.Framework.Graphics; using osu.Framework.Graphics.Containers; using osu.Framework.Logging; using osu.Game.Beatmaps; using osu.Game.Database; using osu.Game.Online.API; using osu.Game.Online.Rooms; using osu.Game.Screens.OnlinePlay; namespace osu.Game.Tests.Visual.OnlinePlay { /// /// A base test scene for all online play components and screens. /// public abstract partial class OnlinePlayTestScene : ScreenTestScene, IOnlinePlayTestSceneDependencies { public Bindable SelectedRoom => OnlinePlayDependencies.SelectedRoom; public IRoomManager RoomManager => OnlinePlayDependencies.RoomManager; public OngoingOperationTracker OngoingOperationTracker => OnlinePlayDependencies.OngoingOperationTracker; public OnlinePlayBeatmapAvailabilityTracker AvailabilityTracker => OnlinePlayDependencies.AvailabilityTracker; public TestUserLookupCache UserLookupCache => OnlinePlayDependencies.UserLookupCache; public BeatmapLookupCache BeatmapLookupCache => OnlinePlayDependencies.BeatmapLookupCache; /// /// All dependencies required for online play components and screens. /// protected OnlinePlayTestSceneDependencies OnlinePlayDependencies => dependencies.OnlinePlayDependencies; protected override Container Content => content; private readonly Container content; private readonly Container drawableDependenciesContainer; private DelegatedDependencyContainer dependencies = null!; protected OnlinePlayTestScene() { base.Content.AddRange(new Drawable[] { drawableDependenciesContainer = new Container { RelativeSizeAxes = Axes.Both }, content = new Container { RelativeSizeAxes = Axes.Both }, }); } protected sealed override IReadOnlyDependencyContainer CreateChildDependencies(IReadOnlyDependencyContainer parent) { return dependencies = new DelegatedDependencyContainer(base.CreateChildDependencies(parent)) { OnlinePlayDependencies = initDependencies() }; } public override void SetUpSteps() { base.SetUpSteps(); AddStep("setup dependencies", () => { // Reset the room dependencies to a fresh state. dependencies.OnlinePlayDependencies = initDependencies(); var handler = OnlinePlayDependencies.RequestsHandler; // Resolving the BeatmapManager in the test scene will inject the game-wide BeatmapManager, while many test scenes cache their own BeatmapManager instead. // To get around this, the BeatmapManager is looked up from the dependencies provided to the children of the test scene instead. var beatmapManager = dependencies.Get(); ((DummyAPIAccess)API).HandleRequest = request => { try { return handler.HandleRequest(request, API.LocalUser.Value, beatmapManager); } catch (ObjectDisposedException) { // These requests can be fired asynchronously, but potentially arrive after game components // have been disposed (ie. realm in BeatmapManager). // This only happens in tests and it's easiest to ignore them for now. Logger.Log($"Handled {nameof(ObjectDisposedException)} in test request handling"); return true; } }; }); } private OnlinePlayTestSceneDependencies initDependencies() { var newDependencies = CreateOnlinePlayDependencies(); drawableDependenciesContainer.Clear(); drawableDependenciesContainer.AddRange(newDependencies.DrawableComponents); return newDependencies; } /// /// Creates the room dependencies. Called every . /// /// /// Any custom dependencies required for online play sub-classes should be added here. /// protected virtual OnlinePlayTestSceneDependencies CreateOnlinePlayDependencies() => new OnlinePlayTestSceneDependencies(); /// /// A providing a mutable lookup source for online play dependencies. /// private class DelegatedDependencyContainer : IReadOnlyDependencyContainer { /// /// The online play dependencies. /// public OnlinePlayTestSceneDependencies OnlinePlayDependencies { get; set; } = null!; private readonly IReadOnlyDependencyContainer parent; private readonly DependencyContainer injectableDependencies; /// /// Creates a new . /// /// The fallback to use when cannot satisfy a dependency. public DelegatedDependencyContainer(IReadOnlyDependencyContainer parent) { this.parent = parent; injectableDependencies = new DependencyContainer(this); } public object Get(Type type) => OnlinePlayDependencies.Get(type) ?? parent.Get(type); public object Get(Type type, CacheInfo info) => OnlinePlayDependencies.Get(type, info) ?? parent.Get(type, info); public void Inject(T instance) where T : class, IDependencyInjectionCandidate => injectableDependencies.Inject(instance); } } }