osu/osu.Game.Tests/Visual/Multiplayer/TestSceneMultiSpectatorScreen.cs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

504 lines
20 KiB
C#
Raw Normal View History

2021-04-08 13:14:30 +00: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;
2021-04-08 13:14:30 +00:00
using System.Collections.Generic;
using System.Linq;
using NUnit.Framework;
using osu.Framework.Allocation;
using osu.Framework.Extensions;
2021-08-25 08:30:37 +00:00
using osu.Framework.Extensions.ObjectExtensions;
2021-04-08 13:14:30 +00:00
using osu.Framework.Graphics;
2021-08-13 03:31:39 +00:00
using osu.Framework.Graphics.Containers;
2021-04-08 13:14:30 +00:00
using osu.Framework.Testing;
using osu.Game.Beatmaps;
using osu.Game.Configuration;
using osu.Game.Online.API.Requests.Responses;
using osu.Game.Online.Multiplayer;
using osu.Game.Online.Multiplayer.MatchTypes.TeamVersus;
2021-06-11 09:13:54 +00:00
using osu.Game.Rulesets.UI;
2021-04-08 13:14:30 +00:00
using osu.Game.Screens.OnlinePlay.Multiplayer.Spectate;
using osu.Game.Screens.Play;
2021-08-13 03:31:39 +00:00
using osu.Game.Screens.Play.HUD;
using osu.Game.Screens.Play.PlayerSettings;
using osu.Game.Storyboards;
2021-04-08 13:14:30 +00:00
using osu.Game.Tests.Beatmaps.IO;
using osuTK;
2021-08-25 08:30:37 +00:00
using osuTK.Graphics;
2021-04-08 13:14:30 +00:00
namespace osu.Game.Tests.Visual.Multiplayer
{
2021-04-23 10:23:52 +00:00
public class TestSceneMultiSpectatorScreen : MultiplayerTestScene
2021-04-08 13:14:30 +00:00
{
[Resolved]
private OsuGameBase game { get; set; } = null!;
2021-04-08 13:14:30 +00:00
[Resolved]
private OsuConfigManager config { get; set; } = null!;
2021-04-08 13:14:30 +00:00
[Resolved]
private BeatmapManager beatmapManager { get; set; } = null!;
2021-04-08 13:14:30 +00:00
private MultiSpectatorScreen spectatorScreen = null!;
2021-04-08 13:14:30 +00:00
private readonly List<MultiplayerRoomUser> playingUsers = new List<MultiplayerRoomUser>();
2021-04-08 13:14:30 +00:00
private BeatmapSetInfo importedSet = null!;
private BeatmapInfo importedBeatmap = null!;
2021-04-08 13:14:30 +00:00
private int importedBeatmapId;
[BackgroundDependencyLoader]
private void load()
{
2021-12-17 09:26:12 +00:00
importedSet = BeatmapImportHelper.LoadOszIntoOsu(game, virtualTrack: true).GetResultSafely();
importedBeatmap = importedSet.Beatmaps.First(b => b.Ruleset.OnlineID == 0);
importedBeatmapId = importedBeatmap.OnlineID;
2021-04-08 13:14:30 +00:00
}
public override void SetUpSteps()
{
base.SetUpSteps();
AddStep("clear playing users", () => playingUsers.Clear());
}
2021-04-08 13:14:30 +00:00
[Test]
public void TestDelayedStart()
{
AddStep("start players silently", () =>
{
OnlinePlayDependencies.MultiplayerClient.AddUser(new APIUser { Id = PLAYER_1_ID }, true);
OnlinePlayDependencies.MultiplayerClient.AddUser(new APIUser { Id = PLAYER_2_ID }, true);
playingUsers.Add(new MultiplayerRoomUser(PLAYER_1_ID));
playingUsers.Add(new MultiplayerRoomUser(PLAYER_2_ID));
});
loadSpectateScreen(false);
AddWaitStep("wait a bit", 10);
AddStep("load player first_player_id", () => SpectatorClient.SendStartPlay(PLAYER_1_ID, importedBeatmapId));
2021-04-22 14:39:02 +00:00
AddUntilStep("one player added", () => spectatorScreen.ChildrenOfType<Player>().Count() == 1);
AddWaitStep("wait a bit", 10);
AddStep("load player second_player_id", () => SpectatorClient.SendStartPlay(PLAYER_2_ID, importedBeatmapId));
2021-04-22 14:39:02 +00:00
AddUntilStep("two players added", () => spectatorScreen.ChildrenOfType<Player>().Count() == 2);
}
2021-04-08 13:14:30 +00:00
[Test]
public void TestGeneral()
{
int[] userIds = getPlayerIds(4);
2021-04-08 13:14:30 +00:00
start(userIds);
loadSpectateScreen();
sendFrames(userIds, 1000);
AddWaitStep("wait a bit", 20);
}
2021-08-13 03:31:39 +00:00
[Test]
public void TestSpectatorPlayerInteractiveElementsHidden()
2021-08-13 03:31:39 +00:00
{
HUDVisibilityMode originalConfigValue = default;
AddStep("get original config hud visibility", () => originalConfigValue = config.Get<HUDVisibilityMode>(OsuSetting.HUDVisibilityMode));
AddStep("set config hud visibility to always", () => config.SetValue(OsuSetting.HUDVisibilityMode, HUDVisibilityMode.Always));
2021-08-13 03:31:39 +00:00
start(new[] { PLAYER_1_ID, PLAYER_2_ID });
loadSpectateScreen(false);
AddUntilStep("wait for player loaders", () => this.ChildrenOfType<PlayerLoader>().Count() == 2);
AddAssert("all player loader settings hidden", () => this.ChildrenOfType<PlayerLoader>().All(l => !l.ChildrenOfType<FillFlowContainer<PlayerSettingsGroup>>().Any()));
2021-08-13 03:31:39 +00:00
AddUntilStep("wait for players to load", () => spectatorScreen.AllPlayersLoaded);
// components wrapped in skinnable target containers load asynchronously, potentially taking more than one frame to load.
// therefore use until step rather than direct assert to account for that.
AddUntilStep("all interactive elements removed", () => this.ChildrenOfType<Player>().All(p =>
!p.ChildrenOfType<PlayerSettingsOverlay>().Any() &&
!p.ChildrenOfType<HoldForMenuButton>().Any() &&
2021-08-13 12:24:10 +00:00
p.ChildrenOfType<SongProgressBar>().SingleOrDefault()?.ShowHandle == false));
AddStep("restore config hud visibility", () => config.SetValue(OsuSetting.HUDVisibilityMode, originalConfigValue));
2021-08-13 03:31:39 +00:00
}
[Test]
public void TestTeamDisplay()
{
AddStep("start players", () =>
{
var player1 = OnlinePlayDependencies.MultiplayerClient.AddUser(new APIUser { Id = PLAYER_1_ID }, true);
player1.MatchState = new TeamVersusUserState
{
TeamID = 0,
};
var player2 = OnlinePlayDependencies.MultiplayerClient.AddUser(new APIUser { Id = PLAYER_2_ID }, true);
player2.MatchState = new TeamVersusUserState
{
TeamID = 1,
};
SpectatorClient.SendStartPlay(player1.UserID, importedBeatmapId);
SpectatorClient.SendStartPlay(player2.UserID, importedBeatmapId);
playingUsers.Add(player1);
playingUsers.Add(player2);
});
loadSpectateScreen();
sendFrames(PLAYER_1_ID, 1000);
sendFrames(PLAYER_2_ID, 1000);
AddWaitStep("wait a bit", 20);
}
2021-06-11 10:15:53 +00:00
[Test]
public void TestTimeDoesNotProgressWhileAllPlayersPaused()
{
start(new[] { PLAYER_1_ID, PLAYER_2_ID });
loadSpectateScreen();
sendFrames(PLAYER_1_ID, 40);
sendFrames(PLAYER_2_ID, 20);
2021-06-11 10:15:53 +00:00
waitUntilPaused(PLAYER_2_ID, true);
2021-06-11 10:15:53 +00:00
checkPausedInstant(PLAYER_1_ID, false);
AddAssert("master clock still running", () => this.ChildrenOfType<MasterGameplayClockContainer>().Single().IsRunning);
waitUntilPaused(PLAYER_1_ID, true);
2021-06-11 10:15:53 +00:00
AddUntilStep("master clock paused", () => !this.ChildrenOfType<MasterGameplayClockContainer>().Single().IsRunning);
}
2021-04-08 13:14:30 +00:00
[Test]
public void TestPlayersMustStartSimultaneously()
{
start(new[] { PLAYER_1_ID, PLAYER_2_ID });
2021-04-08 13:14:30 +00:00
loadSpectateScreen();
// Send frames for one player only, both should remain paused.
sendFrames(PLAYER_1_ID, 20);
checkPausedInstant(PLAYER_1_ID, true);
checkPausedInstant(PLAYER_2_ID, true);
2021-04-08 13:14:30 +00:00
// Send frames for the other player, both should now start playing.
sendFrames(PLAYER_2_ID, 20);
checkPausedInstant(PLAYER_1_ID, false);
checkPausedInstant(PLAYER_2_ID, false);
2021-04-08 13:14:30 +00:00
}
[Test]
public void TestPlayersDoNotStartSimultaneouslyIfBufferingForMaximumStartDelay()
2021-04-08 13:14:30 +00:00
{
start(new[] { PLAYER_1_ID, PLAYER_2_ID });
2021-04-08 13:14:30 +00:00
loadSpectateScreen();
// Send frames for one player only, both should remain paused.
sendFrames(PLAYER_1_ID, 1000);
checkPausedInstant(PLAYER_1_ID, true);
checkPausedInstant(PLAYER_2_ID, true);
2021-04-08 13:14:30 +00:00
// Wait for the start delay seconds...
2022-08-24 06:07:04 +00:00
AddWaitStep("wait maximum start delay seconds", (int)(SpectatorSyncManager.MAXIMUM_START_DELAY / TimePerAction));
2021-04-08 13:14:30 +00:00
// Player 1 should start playing by itself, player 2 should remain paused.
checkPausedInstant(PLAYER_1_ID, false);
checkPausedInstant(PLAYER_2_ID, true);
2021-04-08 13:14:30 +00:00
}
[Test]
public void TestPlayersContinueWhileOthersBuffer()
{
start(new[] { PLAYER_1_ID, PLAYER_2_ID });
2021-04-08 13:14:30 +00:00
loadSpectateScreen();
// Send initial frames for both players. A few more for player 1.
sendFrames(PLAYER_1_ID, 20);
2021-07-05 15:52:39 +00:00
sendFrames(PLAYER_2_ID);
checkPausedInstant(PLAYER_1_ID, false);
checkPausedInstant(PLAYER_2_ID, false);
2021-04-08 13:14:30 +00:00
// Eventually player 2 will pause, player 1 must remain running.
waitUntilPaused(PLAYER_2_ID, true);
checkPausedInstant(PLAYER_1_ID, false);
2021-04-08 13:14:30 +00:00
// Eventually both players will run out of frames and should pause.
waitUntilPaused(PLAYER_1_ID, true);
checkPausedInstant(PLAYER_2_ID, true);
2021-04-08 13:14:30 +00:00
// Send more frames for the first player only. Player 1 should start playing with player 2 remaining paused.
sendFrames(PLAYER_1_ID, 20);
checkPausedInstant(PLAYER_2_ID, true);
checkPausedInstant(PLAYER_1_ID, false);
2021-04-08 13:14:30 +00:00
// Send more frames for the second player. Both should be playing
sendFrames(PLAYER_2_ID, 20);
checkPausedInstant(PLAYER_2_ID, false);
checkPausedInstant(PLAYER_1_ID, false);
2021-04-08 13:14:30 +00:00
}
[Test]
public void TestPlayersCatchUpAfterFallingBehind()
{
start(new[] { PLAYER_1_ID, PLAYER_2_ID });
2021-04-08 13:14:30 +00:00
loadSpectateScreen();
// Send initial frames for both players. A few more for player 1.
sendFrames(PLAYER_1_ID, 1000);
sendFrames(PLAYER_2_ID, 30);
checkPausedInstant(PLAYER_1_ID, false);
checkPausedInstant(PLAYER_2_ID, false);
2021-04-08 13:14:30 +00:00
// Eventually player 2 will run out of frames and should pause.
waitUntilPaused(PLAYER_2_ID, true);
2021-04-08 13:14:30 +00:00
AddWaitStep("wait a few more frames", 10);
// Send more frames for player 2. It should unpause.
sendFrames(PLAYER_2_ID, 1000);
checkPausedInstant(PLAYER_2_ID, false);
2021-04-08 13:14:30 +00:00
// Player 2 should catch up to player 1 after unpausing.
waitForCatchup(PLAYER_2_ID);
2021-04-13 13:40:10 +00:00
AddWaitStep("wait a bit", 10);
2021-04-08 13:14:30 +00:00
}
[Test]
public void TestMostInSyncUserIsAudioSource()
{
start(new[] { PLAYER_1_ID, PLAYER_2_ID });
loadSpectateScreen();
// With no frames, the synchronisation state will be TooFarAhead.
// In this state, all players should be muted.
2021-04-26 10:01:30 +00:00
assertMuted(PLAYER_1_ID, true);
assertMuted(PLAYER_2_ID, true);
// Send frames for both players, with more frames for player 2.
sendFrames(PLAYER_1_ID, 5);
sendFrames(PLAYER_2_ID, 20);
// While both players are running, one of them should be un-muted.
waitUntilPaused(PLAYER_1_ID, false);
assertOnePlayerNotMuted();
// After player 1 runs out of frames, the un-muted player should always be player 2.
waitUntilPaused(PLAYER_1_ID, true);
waitUntilPaused(PLAYER_2_ID, false);
2021-04-26 10:01:30 +00:00
assertMuted(PLAYER_1_ID, true);
assertMuted(PLAYER_2_ID, false);
sendFrames(PLAYER_1_ID, 100);
waitForCatchup(PLAYER_1_ID);
waitUntilPaused(PLAYER_2_ID, true);
2021-04-26 10:01:30 +00:00
assertMuted(PLAYER_1_ID, false);
assertMuted(PLAYER_2_ID, true);
sendFrames(PLAYER_2_ID, 100);
waitForCatchup(PLAYER_2_ID);
2021-04-26 10:01:30 +00:00
assertMuted(PLAYER_1_ID, false);
assertMuted(PLAYER_2_ID, true);
}
2021-06-11 09:13:54 +00:00
[Test]
public void TestSpectatingDuringGameplay()
{
int[] players = { PLAYER_1_ID, PLAYER_2_ID };
2021-06-11 09:13:54 +00:00
start(players);
sendFrames(players, 300);
loadSpectateScreen();
sendFrames(players, 300);
AddUntilStep("playing from correct point in time", () => this.ChildrenOfType<DrawableRuleset>().All(r => r.FrameStableClock.CurrentTime > 30000));
}
[Test]
public void TestSpectatingDuringGameplayWithLateFrames()
{
start(new[] { PLAYER_1_ID, PLAYER_2_ID });
sendFrames(new[] { PLAYER_1_ID, PLAYER_2_ID }, 300);
loadSpectateScreen();
sendFrames(PLAYER_1_ID, 300);
2022-08-24 06:07:04 +00:00
AddWaitStep("wait maximum start delay seconds", (int)(SpectatorSyncManager.MAXIMUM_START_DELAY / TimePerAction));
waitUntilPaused(PLAYER_1_ID, false);
2021-06-11 09:13:54 +00:00
sendFrames(PLAYER_2_ID, 300);
AddUntilStep("player 2 playing from correct point in time", () => getPlayer(PLAYER_2_ID).ChildrenOfType<DrawableRuleset>().Single().FrameStableClock.CurrentTime > 30000);
}
2021-08-25 08:30:37 +00:00
[Test]
public void TestPlayersLeaveWhileSpectating()
{
start(getPlayerIds(4));
sendFrames(getPlayerIds(4), 300);
2021-08-25 08:30:37 +00:00
loadSpectateScreen();
for (int count = 3; count >= 0; count--)
2021-08-25 08:30:37 +00:00
{
int id = PLAYER_1_ID + count;
2021-08-25 08:30:37 +00:00
end(id);
AddUntilStep($"{id} area grayed", () => getInstance(id).Colour != Color4.White);
AddUntilStep($"{id} score quit set", () => getLeaderboardScore(id).HasQuit.Value);
sendFrames(getPlayerIds(count), 300);
2021-08-25 08:30:37 +00:00
}
2021-08-27 10:15:12 +00:00
Player? player = null;
2021-08-27 10:15:12 +00:00
AddStep($"get {PLAYER_1_ID} player instance", () => player = getInstance(PLAYER_1_ID).ChildrenOfType<Player>().Single());
start(new[] { PLAYER_1_ID });
sendFrames(PLAYER_1_ID, 300);
AddAssert($"{PLAYER_1_ID} player instance still same", () => getInstance(PLAYER_1_ID).ChildrenOfType<Player>().Single() == player);
AddAssert($"{PLAYER_1_ID} area still grayed", () => getInstance(PLAYER_1_ID).Colour != Color4.White);
AddAssert($"{PLAYER_1_ID} score quit still set", () => getLeaderboardScore(PLAYER_1_ID).HasQuit.Value);
2021-08-25 08:30:37 +00:00
}
2021-08-11 09:16:25 +00:00
/// <summary>
/// Tests spectating with a beatmap that has a high <see cref="BeatmapInfo.AudioLeadIn"/> value.
///
/// This test is not intended not to check the correct initial time value, but only to guard against
/// gameplay potentially getting stuck in a stopped state due to lead in time being present.
2021-08-11 09:16:25 +00:00
/// </summary>
[Test]
public void TestAudioLeadIn() => testLeadIn(b => b.BeatmapInfo.AudioLeadIn = 2000);
/// <summary>
/// Tests spectating with a beatmap that has a storyboard element with a negative start time (i.e. intro storyboard element).
///
/// This test is not intended not to check the correct initial time value, but only to guard against
/// gameplay potentially getting stuck in a stopped state due to lead in time being present.
/// </summary>
[Test]
public void TestIntroStoryboardElement() => testLeadIn(b =>
{
var sprite = new StoryboardSprite("unknown", Anchor.TopLeft, Vector2.Zero);
sprite.TimelineGroup.Alpha.Add(Easing.None, -2000, 0, 0, 1);
b.Storyboard.GetLayer("Background").Add(sprite);
});
private void testLeadIn(Action<WorkingBeatmap>? applyToBeatmap = null)
2021-08-11 09:16:25 +00:00
{
start(PLAYER_1_ID);
loadSpectateScreen(false, applyToBeatmap);
2021-08-11 09:16:25 +00:00
// to ensure negative gameplay start time does not affect spectator, send frames exactly after StartGameplay().
// (similar to real spectating sessions in which the first frames get sent between StartGameplay() and player load complete)
AddStep("send frames at gameplay start", () => getInstance(PLAYER_1_ID).OnGameplayStarted += () => SpectatorClient.SendFramesFromUser(PLAYER_1_ID, 100));
2021-08-11 09:16:25 +00:00
AddUntilStep("wait for player load", () => spectatorScreen.AllPlayersLoaded);
2022-08-24 08:17:59 +00:00
AddUntilStep("wait for clock running", () => getInstance(PLAYER_1_ID).SpectatorPlayerClock.IsRunning);
2021-08-11 09:16:25 +00:00
assertNotCatchingUp(PLAYER_1_ID);
assertRunning(PLAYER_1_ID);
}
private void loadSpectateScreen(bool waitForPlayerLoad = true, Action<WorkingBeatmap>? applyToBeatmap = null)
2021-04-08 13:14:30 +00:00
{
AddStep("load screen", () =>
2021-04-08 13:14:30 +00:00
{
Beatmap.Value = beatmapManager.GetWorkingBeatmap(importedBeatmap);
Ruleset.Value = importedBeatmap.Ruleset;
applyToBeatmap?.Invoke(Beatmap.Value);
LoadScreen(spectatorScreen = new MultiSpectatorScreen(SelectedRoom.Value, playingUsers.ToArray()));
2021-04-08 13:14:30 +00:00
});
2021-04-22 14:39:02 +00:00
AddUntilStep("wait for screen load", () => spectatorScreen.LoadState == LoadState.Loaded && (!waitForPlayerLoad || spectatorScreen.AllPlayersLoaded));
2021-04-08 13:14:30 +00:00
}
2021-08-11 09:16:25 +00:00
private void start(int userId, int? beatmapId = null) => start(new[] { userId }, beatmapId);
2021-04-08 13:14:30 +00:00
private void start(int[] userIds, int? beatmapId = null)
{
AddStep("start play", () =>
{
foreach (int id in userIds)
{
2021-08-25 08:30:37 +00:00
var user = new MultiplayerRoomUser(id)
{
User = new APIUser { Id = id },
2021-08-25 08:30:37 +00:00
};
OnlinePlayDependencies.MultiplayerClient.AddUser(user.User, true);
SpectatorClient.SendStartPlay(id, beatmapId ?? importedBeatmapId);
2021-08-25 08:30:37 +00:00
playingUsers.Add(user);
}
});
}
private void end(int userId)
2021-08-25 08:30:37 +00:00
{
AddStep($"end play for {userId}", () =>
2021-08-25 08:30:37 +00:00
{
var user = playingUsers.Single(u => u.UserID == userId);
2021-08-25 08:30:37 +00:00
SpectatorClient.SendEndPlay(userId);
OnlinePlayDependencies.MultiplayerClient.RemoveUser(user.User.AsNonNull());
2021-08-25 08:30:37 +00:00
playingUsers.Remove(user);
2021-04-08 13:14:30 +00:00
});
}
/// <summary>
/// Send new frames on behalf of a user.
/// Frames will last for count * 100 milliseconds.
/// </summary>
2021-04-08 13:14:30 +00:00
private void sendFrames(int userId, int count = 10) => sendFrames(new[] { userId }, count);
private void sendFrames(int[] userIds, int count = 10)
{
AddStep("send frames", () =>
{
foreach (int id in userIds)
SpectatorClient.SendFramesFromUser(id, count);
2021-04-08 13:14:30 +00:00
});
}
private void waitUntilPaused(int userId, bool state)
=> AddUntilStep($"{nameof(waitUntilPaused)}({userId}, {state})", () => getPlayer(userId).ChildrenOfType<GameplayClockContainer>().First().IsRunning != state);
private void checkPausedInstant(int userId, bool state)
{
waitUntilPaused(userId, state);
// Todo: The following should work, but is broken because SpectatorScreen retrieves the WorkingBeatmap via the BeatmapManager, bypassing the test scene clock and running real-time.
// AddAssert($"{userId} is {(state ? "paused" : "playing")}", () => getPlayer(userId).ChildrenOfType<GameplayClockContainer>().First().GameplayClock.IsRunning != state);
}
private void assertOnePlayerNotMuted() => AddAssert(nameof(assertOnePlayerNotMuted), () => spectatorScreen.ChildrenOfType<PlayerArea>().Count(p => !p.Mute) == 1);
2021-04-26 10:01:30 +00:00
private void assertMuted(int userId, bool muted)
=> AddAssert($"{nameof(assertMuted)}({userId}, {muted})", () => getInstance(userId).Mute == muted);
2021-04-08 13:14:30 +00:00
2021-08-11 09:16:25 +00:00
private void assertRunning(int userId)
=> AddAssert($"{nameof(assertRunning)}({userId})", () => getInstance(userId).SpectatorPlayerClock.IsRunning);
2021-08-11 09:16:25 +00:00
private void assertNotCatchingUp(int userId)
=> AddAssert($"{nameof(assertNotCatchingUp)}({userId})", () => !getInstance(userId).SpectatorPlayerClock.IsCatchingUp);
2021-08-11 09:16:25 +00:00
private void waitForCatchup(int userId)
=> AddUntilStep($"{nameof(waitForCatchup)}({userId})", () => !getInstance(userId).SpectatorPlayerClock.IsCatchingUp);
2021-04-08 13:14:30 +00:00
private Player getPlayer(int userId) => getInstance(userId).ChildrenOfType<Player>().Single();
2021-04-22 14:52:22 +00:00
private PlayerArea getInstance(int userId) => spectatorScreen.ChildrenOfType<PlayerArea>().Single(p => p.UserId == userId);
2021-08-25 08:30:37 +00:00
private GameplayLeaderboardScore getLeaderboardScore(int userId) => spectatorScreen.ChildrenOfType<GameplayLeaderboardScore>().Single(s => s.User?.Id == userId);
private int[] getPlayerIds(int count) => Enumerable.Range(PLAYER_1_ID, count).ToArray();
2021-04-08 13:14:30 +00:00
}
}