1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-23 15:13:01 +08:00
osu-lazer/osu.Game.Tests/Visual/Gameplay/TestSceneSpectator.cs

321 lines
9.7 KiB
C#
Raw Normal View History

2020-10-26 18:47:39 +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.Collections.Generic;
using System.Linq;
2021-04-01 22:48:26 +08:00
using System.Threading;
using System.Threading.Tasks;
2020-10-26 18:47:39 +08:00
using NUnit.Framework;
2020-10-26 20:17:12 +08:00
using osu.Framework.Allocation;
using osu.Framework.Bindables;
using osu.Framework.Graphics;
using osu.Framework.Screens;
using osu.Framework.Testing;
using osu.Framework.Utils;
using osu.Game.Beatmaps;
2021-04-01 22:48:26 +08:00
using osu.Game.Database;
using osu.Game.Online;
2020-10-26 20:17:12 +08:00
using osu.Game.Online.Spectator;
using osu.Game.Replays.Legacy;
2020-10-27 13:47:15 +08:00
using osu.Game.Rulesets.Osu;
using osu.Game.Rulesets.Osu.Replays;
using osu.Game.Rulesets.UI;
using osu.Game.Scoring;
2020-10-26 18:47:39 +08:00
using osu.Game.Screens.Play;
using osu.Game.Tests.Beatmaps.IO;
2020-10-26 18:47:39 +08:00
using osu.Game.Users;
namespace osu.Game.Tests.Visual.Gameplay
{
public class TestSceneSpectator : ScreenTestScene
{
2020-10-26 20:17:12 +08:00
[Cached(typeof(SpectatorStreamingClient))]
private TestSpectatorStreamingClient testSpectatorStreamingClient = new TestSpectatorStreamingClient();
2020-10-26 18:47:39 +08:00
2021-04-01 22:48:26 +08:00
[Cached(typeof(UserLookupCache))]
private UserLookupCache lookupCache = new TestUserLookupCache();
// used just to show beatmap card for the time being.
protected override bool UseOnlineAPI => true;
2021-04-01 21:08:52 +08:00
private SoloSpectator spectatorScreen;
[Resolved]
private OsuGameBase game { get; set; }
2020-10-27 21:30:45 +08:00
private int nextFrame;
2020-10-28 21:50:57 +08:00
private BeatmapSetInfo importedBeatmap;
private int importedBeatmapId;
public override void SetUpSteps()
{
base.SetUpSteps();
AddStep("reset sent frames", () => nextFrame = 0);
2020-10-28 21:50:57 +08:00
AddStep("import beatmap", () =>
{
importedBeatmap = ImportBeatmapTest.LoadOszIntoOsu(game, virtualTrack: true).Result;
importedBeatmapId = importedBeatmap.Beatmaps.First(b => b.RulesetID == 0).OnlineBeatmapID ?? -1;
});
AddStep("add streaming client", () =>
{
Remove(testSpectatorStreamingClient);
Add(testSpectatorStreamingClient);
});
finish();
}
2020-10-26 18:47:39 +08:00
[Test]
2020-10-29 14:10:11 +08:00
public void TestFrameStarvationAndResume()
2020-10-26 18:47:39 +08:00
{
loadSpectatingScreen();
2021-04-01 21:08:52 +08:00
AddAssert("screen hasn't changed", () => Stack.CurrentScreen is SoloSpectator);
start();
sendFrames();
waitForPlayer();
2020-10-27 13:47:15 +08:00
AddAssert("ensure frames arrived", () => replayHandler.HasFrames);
2021-04-12 17:50:25 +08:00
AddUntilStep("wait for frame starvation", () => replayHandler.WaitingForFrame);
checkPaused(true);
double? pausedTime = null;
AddStep("store time", () => pausedTime = currentFrameStableTime);
sendFrames();
2021-04-12 17:50:25 +08:00
AddUntilStep("wait for frame starvation", () => replayHandler.WaitingForFrame);
checkPaused(true);
AddAssert("time advanced", () => currentFrameStableTime > pausedTime);
2020-10-26 20:17:12 +08:00
}
[Test]
public void TestPlayStartsWithNoFrames()
{
loadSpectatingScreen();
start();
waitForPlayer();
checkPaused(true);
sendFrames(1000); // send enough frames to ensure play won't be paused
checkPaused(false);
}
2020-10-26 20:27:05 +08:00
[Test]
public void TestSpectatingDuringGameplay()
{
start();
loadSpectatingScreen();
AddStep("advance frame count", () => nextFrame = 300);
sendFrames();
waitForPlayer();
AddUntilStep("playing from correct point in time", () => player.ChildrenOfType<DrawableRuleset>().First().FrameStableClock.CurrentTime > 30000);
2020-10-26 20:27:05 +08:00
}
[Test]
public void TestHostRetriesWhileWatching()
2020-10-26 20:27:05 +08:00
{
loadSpectatingScreen();
start();
sendFrames();
2020-10-28 21:50:57 +08:00
waitForPlayer();
Player lastPlayer = null;
AddStep("store first player", () => lastPlayer = player);
start();
sendFrames();
waitForPlayer();
AddAssert("player is different", () => lastPlayer != player);
2020-10-26 20:27:05 +08:00
}
[Test]
public void TestHostFails()
{
loadSpectatingScreen();
start();
waitForPlayer();
checkPaused(true);
finish();
checkPaused(false);
// TODO: should replay until running out of frames then fail
2020-10-26 20:27:05 +08:00
}
[Test]
public void TestStopWatchingDuringPlay()
{
loadSpectatingScreen();
start();
sendFrames();
waitForPlayer();
AddStep("stop spectating", () => (Stack.CurrentScreen as Player)?.Exit());
AddUntilStep("spectating stopped", () => spectatorScreen.GetChildScreen() == null);
}
[Test]
public void TestStopWatchingThenHostRetries()
{
loadSpectatingScreen();
start();
sendFrames();
waitForPlayer();
AddStep("stop spectating", () => (Stack.CurrentScreen as Player)?.Exit());
AddUntilStep("spectating stopped", () => spectatorScreen.GetChildScreen() == null);
// host starts playing a new session
start();
waitForPlayer();
}
[Test]
public void TestWatchingBeatmapThatDoesntExistLocally()
{
loadSpectatingScreen();
start(-1234);
sendFrames();
2021-04-01 21:08:52 +08:00
AddAssert("screen didn't change", () => Stack.CurrentScreen is SoloSpectator);
2020-10-26 20:27:05 +08:00
}
private OsuFramedReplayInputHandler replayHandler =>
(OsuFramedReplayInputHandler)Stack.ChildrenOfType<OsuInputManager>().First().ReplayInputHandler;
private Player player => Stack.CurrentScreen as Player;
private double currentFrameStableTime
=> player.ChildrenOfType<FrameStabilityContainer>().First().FrameStableClock.CurrentTime;
private void waitForPlayer() => AddUntilStep("wait for player", () => Stack.CurrentScreen is Player);
2020-10-28 21:50:57 +08:00
private void start(int? beatmapId = null) => AddStep("start play", () => testSpectatorStreamingClient.StartPlay(beatmapId ?? importedBeatmapId));
private void finish(int? beatmapId = null) => AddStep("end play", () => testSpectatorStreamingClient.EndPlay(beatmapId ?? importedBeatmapId));
private void checkPaused(bool state) =>
AddUntilStep($"game is {(state ? "paused" : "playing")}", () => player.ChildrenOfType<DrawableRuleset>().First().IsPaused.Value == state);
private void sendFrames(int count = 10)
{
AddStep("send frames", () =>
{
testSpectatorStreamingClient.SendFrames(nextFrame, count);
nextFrame += count;
});
}
private void loadSpectatingScreen()
{
2021-04-01 21:08:52 +08:00
AddStep("load screen", () => LoadScreen(spectatorScreen = new SoloSpectator(testSpectatorStreamingClient.StreamingUser)));
AddUntilStep("wait for screen load", () => spectatorScreen.LoadState == LoadState.Loaded);
}
2020-11-02 20:09:47 +08:00
public class TestSpectatorStreamingClient : SpectatorStreamingClient
2020-10-26 20:17:12 +08:00
{
public readonly User StreamingUser = new User { Id = 55, Username = "Test user" };
2020-10-26 20:17:12 +08:00
public new BindableList<int> PlayingUsers => (BindableList<int>)base.PlayingUsers;
2020-10-28 21:50:57 +08:00
private int beatmapId;
public TestSpectatorStreamingClient()
: base(new DevelopmentEndpointConfiguration())
{
}
2020-10-28 21:50:57 +08:00
public void StartPlay(int beatmapId)
{
this.beatmapId = beatmapId;
sendState(beatmapId);
}
2020-10-28 21:50:57 +08:00
public void EndPlay(int beatmapId)
{
((ISpectatorClient)this).UserFinishedPlaying(StreamingUser.Id, new SpectatorState
{
2020-10-28 21:50:57 +08:00
BeatmapID = beatmapId,
RulesetID = 0,
});
sentState = false;
2020-10-26 20:17:12 +08:00
}
private bool sentState;
public void SendFrames(int index, int count)
2020-10-26 20:17:12 +08:00
{
var frames = new List<LegacyReplayFrame>();
for (int i = index; i < index + count; i++)
2020-10-26 20:17:12 +08:00
{
var buttonState = i == index + count - 1 ? ReplayButtonState.None : ReplayButtonState.Left1;
frames.Add(new LegacyReplayFrame(i * 100, RNG.Next(0, 512), RNG.Next(0, 512), buttonState));
}
var bundle = new FrameDataBundle(new ScoreInfo(), frames);
((ISpectatorClient)this).UserSentFrames(StreamingUser.Id, bundle);
if (!sentState)
2020-10-28 21:50:57 +08:00
sendState(beatmapId);
2020-10-26 20:17:12 +08:00
}
public override void WatchUser(int userId)
{
if (sentState)
{
// usually the server would do this.
2020-10-28 21:50:57 +08:00
sendState(beatmapId);
}
base.WatchUser(userId);
}
2020-10-28 21:50:57 +08:00
private void sendState(int beatmapId)
{
sentState = true;
((ISpectatorClient)this).UserBeganPlaying(StreamingUser.Id, new SpectatorState
{
2020-10-28 21:50:57 +08:00
BeatmapID = beatmapId,
RulesetID = 0,
});
}
2020-10-26 18:47:39 +08:00
}
2021-04-01 22:48:26 +08:00
internal class TestUserLookupCache : UserLookupCache
{
protected override Task<User> ComputeValueAsync(int lookup, CancellationToken token = default) => Task.FromResult(new User
{
Id = lookup,
Username = $"User {lookup}"
});
}
2020-10-26 18:47:39 +08:00
}
}